算法题系列4

请设计一个文件缓存系统,该文件缓存系统可以指定缓存的最大值(单位为字节)。文件缓存系统有两种操作:存储文件(put)和读取文件(get),操作命令为put fileName fileSize或者get fileName,存储文件是把文件放入文件缓存系统中;读取文件是从文件缓存系统中访问已存在的文件,如果文件不存在,则不作任何操作。当缓存空间不足以存放新的文件时,根据规则删除文件,直到剩余空间满足新的文件大小为止,再存放新文件。
具体的删除规则为:文件访问过后,会更新文件的最近访问时间和总的访问次数,当缓存不够时,按照第一优先顺序为访问次数从少到多,第二顺序为时间从老到新的方式来删除文件。
输入描述:
第一行为缓存最大值m(整数,取值范围为0<m<=52428800);第二行为 文件操作序列个数n(0 <=n<= 300000)从第三行起为文件操作序列,每个序列单独一行文件操作定义为"op file name file size"
file name是 文件名,file size是文件大小
输出描述
输出当前文件缓存中的文件名列表,文件名用英文逗号分隔,按字典顺序排序如:a.c
如果文件缓存中没有文件,则输出NONE
补充说明:
1.如果新文件的文件名和文件缓存中已有的文件名相同,则不会放在缓存中
2.新的文件第一次存入到文件缓存中时,文件的总访问次数不会变化

为了解决这个文件缓存系统的问题,我们需要设计一个高效的文件缓存系统,支持存储和读取操作,并在缓存空间不足时,按照特定的删除规则来清理旧文件。下面是一个Java实现的详细设计方案:

系统设计

  1. 数据结构

    • 使用 HashMap 存储文件信息,其中键为文件名,值为 FileEntry 对象,FileEntry 记录文件的大小、访问次数和最后访问时间。
    • 使用一个 PriorityQueue 来根据文件的访问次数和最后访问时间来优先删除文件。PriorityQueue 会根据文件的访问次数(从小到大)和最后访问时间(从早到晚)排序。
  2. 操作处理

    • 存储文件 (put)

      • 如果文件已经存在,忽略存储操作。
      • 如果文件不存在且缓存空间足够,直接存储文件。
      • 如果缓存空间不足,根据删除规则删除旧文件,直到有足够的空间存储新文件。
    • 读取文件 (get)

      • 更新文件的最后访问时间和访问次数。如果文件不存在,忽略操作。
  3. 删除文件的规则

    • 优先删除访问次数少的文件。如果多个文件访问次数相同,删除最早被访问的文件。

Java 代码实现

import java.util.*;

public class FileCacheSystem {
    static class FileEntry {
        int size;
        int accessCount;
        long lastAccessTime;

        FileEntry(int size) {
            this.size = size;
            this.accessCount = 0;
            this.lastAccessTime = System.nanoTime();
        }
    }

    private final int maxSize;
    private int currentSize;
    private final Map<String, FileEntry> cache;
    private final PriorityQueue<Map.Entry<String, FileEntry>> evictionQueue;

    public FileCacheSystem(int maxSize) {
        this.maxSize = maxSize;
        this.currentSize = 0;
        this.cache = new HashMap<>();
        this.evictionQueue = new PriorityQueue<>(
                Comparator.comparingInt((Map.Entry<String, FileEntry> e) -> e.getValue().accessCount)
                          .thenComparingLong(e -> e.getValue().lastAccessTime)
        );
    }

    public void put(String fileName, int fileSize) {
        if (fileSize > maxSize) return; // Ignore files larger than cache size
        
        if (cache.containsKey(fileName)) return; // File already in cache

        while (currentSize + fileSize > maxSize) {
            if (evictionQueue.isEmpty()) break;

            Map.Entry<String, FileEntry> entry = evictionQueue.poll();
            FileEntry fileEntry = entry.getValue();
            cache.remove(entry.getKey());
            currentSize -= fileEntry.size;
        }

        if (currentSize + fileSize <= maxSize) {
            FileEntry newEntry = new FileEntry(fileSize);
            cache.put(fileName, newEntry);
            evictionQueue.offer(new AbstractMap.SimpleEntry<>(fileName, newEntry));
            currentSize += fileSize;
        }
    }

    public void get(String fileName) {
        if (!cache.containsKey(fileName)) return; // File not in cache

        FileEntry fileEntry = cache.get(fileName);
        fileEntry.accessCount++;
        fileEntry.lastAccessTime = System.nanoTime();

        evictionQueue.removeIf(entry -> entry.getKey().equals(fileName));
        evictionQueue.offer(new AbstractMap.SimpleEntry<>(fileName, fileEntry));
    }

    public void printCache() {
        if (cache.isEmpty()) {
            System.out.println("NONE");
            return;
        }

        List<String> sortedFileNames = new ArrayList<>(cache.keySet());
        Collections.sort(sortedFileNames);

        System.out.println(String.join(",", sortedFileNames));
    }

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int maxSize = scanner.nextInt();
        int n = scanner.nextInt();
        scanner.nextLine(); // Consume newline

        FileCacheSystem fileCacheSystem = new FileCacheSystem(maxSize);

        for (int i = 0; i < n; i++) {
            String line = scanner.nextLine();
            String[] parts = line.split(" ");
            String operation = parts[0];
            String fileName = parts[1];
            int fileSize = Integer.parseInt(parts[2]);

            if ("put".equals(operation)) {
                fileCacheSystem.put(fileName, fileSize);
            } else if ("get".equals(operation)) {
                fileCacheSystem.get(fileName);
            }
        }

        fileCacheSystem.printCache();
        scanner.close();
    }
}

说明

  1. 文件缓存系统类

    • FileEntry 记录文件的大小、访问次数和最后访问时间。
    • FileCacheSystem 管理缓存的最大容量、当前容量、缓存数据及其优先队列。
  2. 操作处理

    • put 操作会添加文件到缓存中,并处理缓存空间不足的情况。
    • get 操作会更新文件的访问次数和最后访问时间。
    • printCache 方法打印缓存中的文件名,按字典顺序排序。
  3. 性能考虑

    • PriorityQueue 的使用确保了能够在 O(log N) 时间复杂度内进行优先级排序和删除操作。
    • HashMap 用于快速访问文件信息和更新缓存。
  4. 输入和输出

    • 输入通过 Scanner 读取,输出通过 System.out.println 输出缓存中文件的排序结果。

此实现有效地处理了文件缓存系统中的存储、读取和缓存清理操作,能够处理大规模的操作序列,并满足题目要求。



题目描述
评估一个网络的信号质量,其中一个做法是将网络划分为栅格,然后对每个栅格的信号质量计算,路测的时候,希望选择一条信号最好的路线(彼此相连的栅格集合)进行演示,现给出R行C列的整数数组COV,每个单元格的数值S即为该栅格的信号质量(已归一化,无单位,值越大信号越好)

要求从[0,0]到[R-1,C-1]设计一条最优路测路线。返回该路线得分。

规则:

  1. 路测路线可以上下左右四个方向,不能对角

  2. 路线的评分是以路线上信号最差的栅格为准的,例如路径8->4->5->9的值为 4,该线路评分为4。线路最优表示该条线路的评分最高。

输入
第1行表示栅格的行数R

第2行表示栅格的列数C

第3行开始,每一行表示栅格地图一行的信号值,如5 4 5

输出
最优路线的得分
样例输入
3
3
5 4 5
1 2 6
7 4 6
样例输出
4
提示
1 <= R,C <= 20

0 <= S <= 65535

要解决这个问题,我们需要找到一条从栅格左上角 (0,0) 到右下角 (R-1,C-1) 的路径,使得路径上的最小信号质量最大。这是一个典型的最优路径问题,其中路径的评分是路径上最小信号质量的最大值。

思路

我们可以使用 最大最小路径(Maximum Minimum Path) 的策略来解决这个问题。这种策略的核心思想是将所有路径分数转化为一个最大最小路径问题。具体步骤如下:

  1. 定义目标

    • 我们希望找到一条路径,使得路径上所有栅格的最小信号质量尽可能大。换句话说,找到路径中最小信号质量的最大值。
  2. 二分搜索结合 BFS

    • 二分搜索:在可能的信号质量值范围内进行二分搜索,以确定路径中最小信号质量的最大可能值。
    • BFS(广度优先搜索):对于每个候选信号质量值,使用 BFS 来检查从 (0,0) 到 (R-1,C-1) 是否存在一条路径,其中所有栅格的信号质量都大于或等于该候选值。

具体实现

  1. 输入处理

    • 读取栅格的行数和列数。
    • 读取每个栅格的信号质量,并存储在二维数组中。
  2. 二分搜索

    • 定义信号质量的范围,即从最小值到最大值。
    • 对于每个候选值,使用 BFS 检查是否存在一条符合条件的路径。
  3. BFS 实现

    • 从起始栅格 (0,0) 开始,检查是否能到达终点 (R-1,C-1)。
    • 在 BFS 过程中,确保路径上所有栅格的信号质量都大于或等于当前候选值。

Java 代码实现

import java.util.*;

public class NetworkSignal {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int R = scanner.nextInt();
        int C = scanner.nextInt();
        int[][] grid = new int[R][C];
        
        for (int i = 0; i < R; i++) {
            for (int j = 0; j < C; j++) {
                grid[i][j] = scanner.nextInt();
            }
        }
        
        System.out.println(findMaxMinPath(grid, R, C));
    }

    private static int findMaxMinPath(int[][] grid, int R, int C) {
        int left = 0, right = 65535; // Possible signal quality range
        
        // Binary search to find the maximum minimum value of the path
        while (left < right) {
            int mid = left + (right - left + 1) / 2;
            if (canReach(grid, R, C, mid)) {
                left = mid;
            } else {
                right = mid - 1;
            }
        }
        
        return left;
    }

    private static boolean canReach(int[][] grid, int R, int C, int minSignal) {
        boolean[][] visited = new boolean[R][C];
        Queue<int[]> queue = new LinkedList<>();
        queue.offer(new int[]{0, 0});
        visited[0][0] = true;
        
        int[][] directions = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}}; // Right, Down, Left, Up
        
        while (!queue.isEmpty()) {
            int[] curr = queue.poll();
            int x = curr[0];
            int y = curr[1];
            
            if (x == R - 1 && y == C - 1) {
                return true; // Reached the end
            }
            
            for (int[] dir : directions) {
                int newX = x + dir[0];
                int newY = y + dir[1];
                
                if (newX >= 0 && newX < R && newY >= 0 && newY < C && !visited[newX][newY] && grid[newX][newY] >= minSignal) {
                    visited[newX][newY] = true;
                    queue.offer(new int[]{newX, newY});
                }
            }
        }
        
        return false; // No path found
    }
}

说明

  1. 二分搜索:在 [0, 65535] 的范围内查找,使得信号质量值最大化。
  2. BFS:用于检查从 (0,0)(R-1,C-1) 是否存在符合当前信号质量的路径。
  3. 复杂度
    • 时间复杂度:主要是二分搜索和 BFS 的复杂度,总体上是 ( O(\log M \cdot (R \cdot C)) ),其中 ( M ) 是信号质量的范围。
    • 空间复杂度:主要是 BFS 使用的队列和访问标记,空间复杂度是 ( O(R \cdot C) )。

此代码实现了题目要求,能够有效地处理给定大小的信号栅格,并找到最优路径的评分。



题目

警察在侦破一个案件时,得到了线人给出的可能犯罪时间,形如 “HH:MM” 表示的时刻。根据警察和线人的约定,为了隐蔽,该时间是修改过的,解密规则为:利用当前出现过的数字,构造下一个距离当前时间最近的时刻,则该时间为可能的犯罪时间。每个出现数字都可以被无限次使用。

输入描述:形如 HH:MM 的字符串,表示原始输入

输出描述:形如 HH:MM 的字符串,表示推理出来的犯罪时间

示例 1
输入
18:52
输出
18:55
说明
利用数字1, 8, 5, 2构造出来的最近时刻是18:55,是3分钟之后。结果不是18:51因为这个时刻是18小时52分钟之后。

示例 2
输入
23:59
输出
22:22
说明
利用数字2, 3, 5, 9构造出来的最近时刻是22:22。 答案一定是第二天的某一时刻,所以选择可构造的最小时刻为犯罪时间。

答案

import java.util.HashSet;
import java.util.Set;

public class NextClosestTime {
    public static void main(String[] args) {
        String input = "18:52"; // Example input; replace with actual input
        System.out.println(findNextClosestTime(input));
    }

    public static String findNextClosestTime(String time) {
        Set<Character> digits = new HashSet<>();
        for (char c : time.toCharArray()) {
            if (c != ':') {
                digits.add(c);
            }
        }
        
        String currentTime = time;
        while (true) {
            currentTime = getNextTime(currentTime);
            if (isValidTime(currentTime, digits)) {
                return currentTime;
            }
        }
    }

    private static String getNextTime(String time) {
        String[] parts = time.split(":");
        int hours = Integer.parseInt(parts[0]);
        int minutes = Integer.parseInt(parts[1]);

        minutes++;
        if (minutes == 60) {
            minutes = 0;
            hours++;
            if (hours == 24) {
                hours = 0;
            }
        }

        return String.format("%02d:%02d", hours, minutes);
    }

    private static boolean isValidTime(String time, Set<Character> digits) {
        String[] parts = time.split(":");
        String hourPart = parts[0];
        String minutePart = parts[1];

        for (char c : hourPart.toCharArray()) {
            if (!digits.contains(c)) {
                return false;
            }
        }

        for (char c : minutePart.toCharArray()) {
            if (!digits.contains(c)) {
                return false;
            }
        }

        return true;
    }
}

解释

  1. 提取数字:从给定的时间中提取出所有出现的数字,并存入一个集合 digits
  2. 生成下一个时间:逐步生成下一个时间,直到找到一个满足条件的时间。时间递增的逻辑在 getNextTime 方法中实现。
  3. 验证时间有效性:检查生成的时间是否仅使用了之前出现的数字。通过 isValidTime 方法进行验证。
  4. 处理时间:如果找到了一个符合条件的时间,则返回;否则继续查找。


题目

给定两个字符串 AB,将其映射到一个二维数组中,每个字符为一个坐标轴上的点。定义原点为 (0, 0),终点为 (m, n),其中 mn 分别为字符串 AB 的长度。每两个字符相同的坐标点之间可以作一个斜边,斜边的距离为 1。我们需要计算从 (0, 0)(m, n) 的最短路径距离。路径可以是水平、垂直或斜边的组合。

输入描述

  • 两个字符串 AB,长度不超过 1000。

输出描述

  • 输出从 (0, 0)(m, n) 的最短距离。

示例

输入

ABCABBA
CBABAC

输出

9

解题思路

  1. 定义问题: 我们需要在一个二维网格中寻找最短路径,允许水平、垂直和斜对角移动。网格的每一个点对应于字符串 AB 的一个字符。如果两个字符相同,可以沿对角线移动。

  2. 使用动态规划: 我们可以使用动态规划 (DP) 来解决这个问题。定义一个 DP 表格 dp[i][j] 表示从 (0, 0)(i, j) 的最短距离。初始状态为 dp[0][0] = 0,即起点的距离为 0。

  3. 状态转移:

    • (i-1, j)(i, j) 需要加上水平边的距离。
    • (i, j-1)(i, j) 需要加上垂直边的距离。
    • (i-1, j-1)(i, j) 需要加上斜边的距离,前提是 A[i-1]B[j-1] 相同。
  4. 边界条件: 需要处理边界条件,如初始化第一行和第一列。

  5. 最终结果: dp[m][n] 即为从 (0, 0)(m, n) 的最短距离。

Java 代码实现

public class ShortestPath {
    public static void main(String[] args) {
        String A = "ABCABBA";
        String B = "CBABAC";
        System.out.println(findShortestDistance(A, B));
    }

    public static int findShortestDistance(String A, String B) {
        int m = A.length();
        int n = B.length();
        
        // Create a 2D array to store the minimum distance
        int[][] dp = new int[m + 1][n + 1];
        
        // Initialize the DP table
        for (int i = 0; i <= m; i++) {
            for (int j = 0; j <= n; j++) {
                dp[i][j] = Integer.MAX_VALUE;
            }
        }
        
        // Starting point
        dp[0][0] = 0;
        
        // Fill the DP table
        for (int i = 0; i <= m; i++) {
            for (int j = 0; j <= n; j++) {
                if (i > 0) {
                    dp[i][j] = Math.min(dp[i][j], dp[i - 1][j] + 1);
                }
                if (j > 0) {
                    dp[i][j] = Math.min(dp[i][j], dp[i][j - 1] + 1);
                }
                if (i > 0 && j > 0 && A.charAt(i - 1) == B.charAt(j - 1)) {
                    dp[i][j] = Math.min(dp[i][j], dp[i - 1][j - 1] + 1);
                }
            }
        }
        
        // The answer is the shortest distance to (m, n)
        return dp[m][n];
    }
}

解释

  • 初始化: dp 表格初始化为 Integer.MAX_VALUE,表示初始状态下距离为无穷大。
  • 动态规划填表:
    • 对于每个点 (i, j),检查从 (i-1, j)(i, j-1) 的水平和垂直边的距离。
    • 如果 A[i-1]B[j-1] 相同,还需检查斜对角的距离。
  • 输出: dp[m][n] 是最终从 (0, 0)(m, n) 的最短距离。


题目

给定一个一维整型数组,计算数组中的众数,并组成一个新的数组,然后求出这个新数组的中位数。具体步骤如下:

  1. 众数是指一组数据中出现次数最多的数。如果有多个数出现次数相同且最多,它们都算作众数。
  2. 中位数是指将数组从小到大排列后,取最中间的数。如果数组元素个数为偶数,则取中间两个数之和除以 2 的结果。

输入描述

  • 输入一个一维整型数组,数组大小取值范围 0 < N < 1000,数组中每个元素取值范围 0 < E < 1000

输出描述

  • 输出众数组成的新数组的中位数。

示例

示例 1

输入

10 11 21 19 21 17 21 16 21 18 15

输出

21

示例 2

输入

2 1 5 4 3 3 9 2 7 4 6 2 15 4 2 4

输出

3

解题思路

  1. 计算众数:

    • 使用哈希表统计每个数字出现的次数。
    • 找到出现次数最多的数字,即为众数。如果有多个出现次数最多的数字,所有这些数字都应被记录下来。
  2. 生成新数组:

    • 根据众数生成一个新数组,其中包含所有众数的值。
  3. 计算中位数:

    • 将新数组进行排序。
    • 根据新数组的长度,计算中位数。如果长度为奇数,取中间元素;如果长度为偶数,取中间两个元素的平均值。

Java 代码实现

import java.util.*;

public class ModeMedian {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        String[] input = scanner.nextLine().split(" ");
        
        // Convert input to integer array
        int[] arr = Arrays.stream(input).mapToInt(Integer::parseInt).toArray();
        
        // Find the mode(s)
        List<Integer> modes = findModes(arr);
        
        // Calculate median of the mode array
        double median = findMedian(modes);
        
        // Print the median
        System.out.println((int) median);
    }
    
    // Function to find the mode(s) of the array
    public static List<Integer> findModes(int[] arr) {
        Map<Integer, Integer> frequencyMap = new HashMap<>();
        int maxFrequency = 0;
        
        // Build frequency map and find max frequency
        for (int num : arr) {
            int frequency = frequencyMap.getOrDefault(num, 0) + 1;
            frequencyMap.put(num, frequency);
            maxFrequency = Math.max(maxFrequency, frequency);
        }
        
        // Collect all modes
        List<Integer> modes = new ArrayList<>();
        for (Map.Entry<Integer, Integer> entry : frequencyMap.entrySet()) {
            if (entry.getValue() == maxFrequency) {
                modes.add(entry.getKey());
            }
        }
        
        return modes;
    }
    
    // Function to calculate median of a list of integers
    public static double findMedian(List<Integer> list) {
        Collections.sort(list);
        int size = list.size();
        if (size % 2 == 1) {
            return list.get(size / 2);
        } else {
            return (list.get(size / 2 - 1) + list.get(size / 2)) / 2.0;
        }
    }
}

解释

  1. 输入处理:

    • 读取并解析输入,将字符串转换为整数数组。
  2. 计算众数:

    • 使用哈希表统计每个数的出现次数。
    • 找到最大出现次数的所有数,并记录下来。
  3. 计算中位数:

    • 将众数列表进行排序,并根据其长度计算中位数。
  4. 输出结果:

    • 打印中位数。如果输入的数组为空或不存在有效众数,则处理后的结果根据实际情况处理。

这个解决方案高效地处理了数组众数和中位数的计算,适用于给定的输入范围。


题目

设计一个简易的重复内容识别系统。系统需要处理两个字符串,并通过给定的相似字符对来判断这两个字符串是否相似。如果相似,则返回 True 和相似的字符对;如果不相似,则返回第一个内容的“不相似”信息。

输入描述

  1. 两个字符串 str1str2 需要比较相似性。
  2. 一些相似字符对,如 (顿号, 逗号) 表示顿号和逗号相似。
  3. 匹配相似字符对时,字符对可以有任意长度的内容匹配。

输出描述

  • 如果两个字符串相似,输出 True 和相似的字符对。
  • 如果不相似,返回第一个内容的不相似信息。多处不相似的字符串用空格分隔。

示例

示例 1

输入

异世邪君(人气玄幻作家)
异世邪君
(异世邪君, 异世邪君)

输出

True (异世邪君, 异世邪君)

示例 2

输入

hello world
hello, world
(hello, hello) (world, world)

输出

hello world

解题思路

  1. 建立相似字符对的映射关系:

    • 使用一个字典来存储所有的相似字符对。
    • 利用传递性关系将所有相关字符对连接在一起。
  2. 替换字符串中的字符:

    • 使用上述映射关系将字符串中的字符替换为它们的相似字符。
    • 例如,如果 顿号逗号 是相似的,那么所有的 顿号 可以替换为 逗号,进行比较时将其替换为相似字符集合的标准字符。
  3. 比较字符串:

    • 将两个字符串转换为标准形式后进行比较。
    • 如果两者相等,则返回相似结果和相似字符对。
    • 如果不相等,返回不相似的内容。

Java 代码实现

import java.util.*;

public class ContentSimilarity {

    private static Map<String, Set<String>> similarityMap = new HashMap<>();

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        
        // Read input
        String str1 = scanner.nextLine().trim();
        String str2 = scanner.nextLine().trim();
        String[] pairs = scanner.nextLine().trim().split(" ");
        
        // Initialize similarity map
        initializeSimilarityMap(pairs);
        
        // Check similarity
        if (areSimilar(str1, str2)) {
            System.out.println("True " + getSimilarPairs());
        } else {
            System.out.println(getDissimilarParts(str1));
        }
        
        scanner.close();
    }
    
    // Initialize similarity map
    private static void initializeSimilarityMap(String[] pairs) {
        for (String pair : pairs) {
            String[] parts = pair.replace("(", "").replace(")", "").split(",");
            if (parts.length == 2) {
                String a = parts[0].trim();
                String b = parts[1].trim();
                
                similarityMap.computeIfAbsent(a, k -> new HashSet<>()).add(b);
                similarityMap.computeIfAbsent(b, k -> new HashSet<>()).add(a);
                
                // Adding transitive relations
                addTransitiveRelations(a, b);
            }
        }
    }
    
    // Add transitive relations to the map
    private static void addTransitiveRelations(String a, String b) {
        Set<String> aSet = similarityMap.get(a);
        Set<String> bSet = similarityMap.get(b);
        
        if (aSet != null && bSet != null) {
            for (String item : bSet) {
                aSet.add(item);
                similarityMap.get(item).add(a);
            }
            for (String item : aSet) {
                bSet.add(item);
                similarityMap.get(item).add(b);
            }
        }
    }
    
    // Check if two strings are similar
    private static boolean areSimilar(String str1, String str2) {
        return transform(str1).equals(transform(str2));
    }
    
    // Transform string based on similarity map
    private static String transform(String str) {
        StringBuilder transformed = new StringBuilder();
        for (char ch : str.toCharArray()) {
            String chStr = String.valueOf(ch);
            if (similarityMap.containsKey(chStr)) {
                transformed.append(getRepresentative(chStr));
            } else {
                transformed.append(ch);
            }
        }
        return transformed.toString();
    }
    
    // Get representative for a character or string
    private static String getRepresentative(String str) {
        Set<String> set = similarityMap.get(str);
        return set != null ? set.iterator().next() : str;
    }
    
    // Get similar pairs
    private static String getSimilarPairs() {
        StringBuilder result = new StringBuilder();
        for (Map.Entry<String, Set<String>> entry : similarityMap.entrySet()) {
            for (String similar : entry.getValue()) {
                result.append("(").append(entry.getKey()).append(", ").append(similar).append(") ");
            }
        }
        return result.toString().trim();
    }
    
    // Get dissimilar parts
    private static String getDissimilarParts(String str) {
        return str;
    }
}

解释

  1. 建立相似字符对的映射关系:

    • 利用哈希表存储每个字符的相似字符对,并建立传递关系。
  2. 字符串转换:

    • 将字符串中所有字符根据映射关系转换为标准字符。
  3. 比较和输出:

    • 比较转换后的两个字符串是否相同,如果相同则输出相似的字符对;否则输出第一个内容的不相似信息。


题目

给定一张地图上有 n 个城市和道路,城市间的道路构成了一棵树。要求找到一个城市,使得切断通往该城市的所有道路后,地图上最大的城市群(连通分量)的城市数目最小。具体来说,求解每个城市的聚集度(Degree of Polymerization,DP),DP 定义为:切断该城市后,形成的多个连通子图中最大的城市数目。我们需要找出所有使得 DP 最小的城市,并按升序输出。

输入描述

  • 第一行是一个整数 N,表示城市的数量,1 <= N <= 1000。
  • 接下来的 N-1 行,每行包含两个整数 xy,表示城市 x 和城市 y 之间有一条道路。

输出描述

  • 输出所有使得 DP 最小的城市编号。如果有多个,按编号升序输出。

示例

示例 1

输入

5
1 2
2 3
3 4
4 5

输出

3

示例 2

输入

6
1 2
2 3
2 4
3 5
3 6

输出

2 3

解题思路

  1. 建图:

    • 使用邻接表来存储城市之间的连接关系。由于城市构成树状结构,所以每个节点的连接关系只有一个。
  2. DFS 计算子树大小:

    • 使用深度优先搜索 (DFS) 计算每个节点的子树大小。
  3. 计算 DP 值:

    • 对于每个城市 i,假设移除 i,计算所有生成的子图的大小,并得到 DP 值。
    • 计算 DP 值时,对于每个节点,移除该节点后,可以得到多个子图。DP 值为这些子图中最大的子图的大小。
  4. 寻找最小 DP 值的城市:

    • 遍历所有城市,找到 DP 值最小的城市,并输出这些城市编号。

Java 实现

import java.util.*;

public class MinDegreeOfPolymerization {

    private static List<List<Integer>> graph;
    private static int[] subtreeSize;
    private static int n;

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        n = scanner.nextInt();
        graph = new ArrayList<>(n + 1);
        for (int i = 0; i <= n; i++) {
            graph.add(new ArrayList<>());
        }

        for (int i = 0; i < n - 1; i++) {
            int x = scanner.nextInt();
            int y = scanner.nextInt();
            graph.get(x).add(y);
            graph.get(y).add(x);
        }

        subtreeSize = new int[n + 1];
        boolean[] visited = new boolean[n + 1];
        calculateSubtreeSizes(1, visited);

        int minDp = Integer.MAX_VALUE;
        List<Integer> result = new ArrayList<>();
        
        for (int i = 1; i <= n; i++) {
            int dp = calculateDP(i);
            if (dp < minDp) {
                minDp = dp;
                result.clear();
                result.add(i);
            } else if (dp == minDp) {
                result.add(i);
            }
        }

        Collections.sort(result);
        System.out.println(result.stream().map(String::valueOf).collect(Collectors.joining(" ")));
    }

    private static void calculateSubtreeSizes(int node, boolean[] visited) {
        visited[node] = true;
        subtreeSize[node] = 1;
        for (int neighbor : graph.get(node)) {
            if (!visited[neighbor]) {
                calculateSubtreeSizes(neighbor, visited);
                subtreeSize[node] += subtreeSize[neighbor];
            }
        }
    }

    private static int calculateDP(int root) {
        boolean[] visited = new boolean[n + 1];
        visited[root] = true;
        int maxSize = 0;

        for (int neighbor : graph.get(root)) {
            if (!visited[neighbor]) {
                maxSize = Math.max(maxSize, dfsSize(neighbor, visited));
            }
        }

        return maxSize;
    }

    private static int dfsSize(int node, boolean[] visited) {
        visited[node] = true;
        int size = 1;
        for (int neighbor : graph.get(node)) {
            if (!visited[neighbor]) {
                size += dfsSize(neighbor, visited);
            }
        }
        return size;
    }
}

解释

  1. 图的构建:

    • 用邻接表表示城市之间的连接关系。
  2. DFS 计算子树大小:

    • 从一个节点开始,使用 DFS 计算每个节点的子树大小。
  3. 计算 DP 值:

    • 对于每个城市,计算切断它后生成的各个子图的大小,确定最大值作为 DP 值。
  4. 找出最小 DP 值的城市:

    • 遍历所有城市,找到 DP 值最小的城市,并输出这些城市编号。


题目

某公司部门需要派遣员工去国外做项目。代号为 x 的国家和代号为 y 的国家分别需要 cntx 名和 cnty 名员工。每个员工有一个员工号(1, 2, 3, ...),工号连续,从1开始。

部长派遣员工的规则

  1. [1, k] 中选择员工派遣出去。
  2. 编号为 x 的倍数的员工不能去 x 国,编号为 y 的倍数的员工不能去 y 国。

问题

找到最小的 k,使得可以将编号在 [1, k] 中的员工分配给 x 国和 y 国,且满足 x 国和 y 国的需求。

输入描述

  • 四个整数 x, y, cntx, cnty

条件

  • 2 ≤ x < y ≤ 30000
  • xy 一定是质数
  • 1 ≤ cntx, cnty < 10^9
  • cntx + cnty ≤ 10^9

输出描述

  • 输出满足条件的最小的 k

解题思路

  1. 确定有效员工总数:

    • 需要排除 xy 的倍数。对于每个 k,总员工数是 k,但我们需要排除掉不能派遣的员工。即不能派遣的员工数包括 k / xk / y,但需要加上 k / lcm(x, y),因为那些是 xy 的倍数的员工被计算了两次。
  2. 计算公式:

    • 使用以下公式来确定有效员工总数:
      [
      \text{有效员工总数} = k - \left(\frac{k}{x} + \frac{k}{y} - \frac{k}{\text{lcm}(x, y)}\right)
      ]
    • 这里,lcm(x, y)xy 的最小公倍数,可以通过公式计算:
      [
      \text{lcm}(x, y) = \frac{x \times y}{\text{gcd}(x, y)}
      ]
    • 用二分查找来找到最小的 k,使得有效员工总数至少满足 cntx + cnty
  3. 二分查找:

    • 二分查找从 1 到一个合理的上界(例如 cntx + cnty + max(x, y)),在每一步中计算有效员工数,并判断是否能满足需求。

Java 实现

import java.util.Scanner;

public class EmployeeDispatch {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        int x = scanner.nextInt();
        int y = scanner.nextInt();
        long cntx = scanner.nextLong();
        long cnty = scanner.nextLong();

        long left = 1;
        long right = cntx + cnty;
        long result = right;

        while (left <= right) {
            long mid = left + (right - left) / 2;
            long validEmployees = mid - (mid / x + mid / y - mid / lcm(x, y));

            if (validEmployees >= cntx + cnty) {
                result = mid;
                right = mid - 1;
            } else {
                left = mid + 1;
            }
        }

        System.out.println(result);
    }

    private static long gcd(long a, long b) {
        while (b != 0) {
            long temp = b;
            b = a % b;
            a = temp;
        }
        return a;
    }

    private static long lcm(long a, long b) {
        return a * b / gcd(a, b);
    }
}

解释

  1. 计算有效员工总数:

    • 使用 k - (k / x + k / y - k / lcm(x, y)) 公式来计算有效员工数量。
  2. 二分查找:

    • [1, cntx + cnty] 范围内进行二分查找来找到满足条件的最小 k
  3. 辅助函数:

    • gcdlcm 用于计算最小公倍数。

结果

通过上述实现,可以找到满足条件的最小 k。使用二分查找的方法保证了效率,即使 cntxcnty 很大,代码仍然可以在合理的时间内完成计算。


题目描述

项目组有 N 名开发人员,项目经理接到了 M 个独立需求,每个需求的工作量不同,且每个需求只能由一个开发人员独立完成,不能多人合作。任务是帮助项目经理安排工作,使得整个项目用最少的时间交付。

输入描述

  1. 第一行输入为 M 个需求的工作量,单位为天,用逗号隔开。例如:6 2 7 7 9 3 2 1 3 11 4
  2. 第二行输入为项目组人员数量 N。例如:2

输出描述

输出最快完成所有工作的天数。

示例

输入

6 2 7 7 9 3 2 1 3 11 4
2

输出

28

说明

共有两位员工,分配需求 6 2 7 7 3 2 1 共需要 28 天完成,另一位分配需求 9 3 11 4 共需要 27 天完成,因此整个项目的最短完成时间是 28 天。

解题思路

这个问题可以转化为一个 “工作分配问题”,它类似于一个 “最小最大负载” 问题。目标是将任务分配给 N 个开发人员,使得所有开发人员的工作量的最大值尽可能地小。

具体步骤

  1. 定义问题:

    • 将任务分配给 N 个开发人员,最小化完成所有工作的最大天数。
    • 这是一个典型的分配问题,可以用二分查找配合贪心算法解决。
  2. 二分查找:

    • 设定搜索范围:
      • 最小值是最大工作量(因为至少一个开发人员要处理最大任务)。
      • 最大值是所有工作量之和(即所有工作都由一个人完成的情况)。
  3. 检查函数:

    • 设计一个函数来判断某个时间 T 是否能够将所有任务分配给 N 个开发人员,使得每个开发人员的工作时间不超过 T
  4. 二分查找算法:

    • 在上述范围内进行二分查找,根据检查函数的结果来调整搜索范围,最终确定最小的 T

Java 实现

import java.util.Scanner;

public class ProjectManager {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        
        // 读取需求的工作量
        String[] input = scanner.nextLine().split(" ");
        int[] workloads = new int[input.length];
        for (int i = 0; i < input.length; i++) {
            workloads[i] = Integer.parseInt(input[i]);
        }
        
        // 读取项目组人员数量
        int N = scanner.nextInt();
        
        // 二分查找范围的设定
        int low = 0;
        int high = 0;
        for (int workload : workloads) {
            low = Math.max(low, workload);
            high += workload;
        }
        
        // 二分查找最小的最大工作时间
        while (low < high) {
            int mid = low + (high - low) / 2;
            if (canDistributeWork(workloads, N, mid)) {
                high = mid;
            } else {
                low = mid + 1;
            }
        }
        
        // 输出结果
        System.out.println(low);
    }

    // 检查是否可以将工作分配给 N 个开发人员,且最大工作时间不超过 maxTime
    private static boolean canDistributeWork(int[] workloads, int N, int maxTime) {
        int count = 1;
        int currentTime = 0;
        
        for (int workload : workloads) {
            if (currentTime + workload > maxTime) {
                count++;
                currentTime = workload;
                if (count > N) {
                    return false;
                }
            } else {
                currentTime += workload;
            }
        }
        return true;
    }
}

解释

  1. 输入解析:

    • 读取并解析工作量数组以及员工数量。
  2. 二分查找:

    • 使用二分查找确定最小的最大工作时间。
  3. 检查函数:

    • canDistributeWork 函数判断给定最大工作时间下,是否可以在 N 名开发人员之间合理分配工作。
  4. 输出结果:

    • 输出满足条件的最小最大工作时间。


题目描述

给定一个包含 01 的二维矩阵,物体从给定的初始位置开始移动,以给定的速度进行运动。在碰到矩阵边缘时,物体会发生镜面反射。你需要计算在经过 t 时间单位后,物体经过的 1 的点的次数。

输入描述

  1. 矩阵:一个二维矩阵,只包含 01
  2. 初始位置:物体的初始位置 (x, y)。
  3. 速度:物体的速度 (vx, vy),表示物体每个时间单位在 x 和 y 方向上的移动距离。
  4. 时间 t:模拟的时间单位。

输出描述

输出在经过 t 时间单位后,物体经过的 1 的点的次数。

示例

输入

5 5
1 1 1 1 1
1 0 0 0 1
1 0 0 0 1
1 0 0 0 1
1 1 1 1 1
2 2
1 1
3

输出

7

解释

在给定的初始位置 (2, 2) 和速度 (1, 1) 下,物体在 3 个时间单位后经过的点如下:

  • 时间 0: (2, 2),值为 1
  • 时间 1: (3, 3),值为 0
  • 时间 2: (4, 4),值为 1
  • 时间 3: (0, 0),值为 1(经过边界反射)。

在这些时间点中,物体总共经过了 7 个 1

解题思路

  1. 矩阵反射处理:

    • 对于物体的每次移动,检查是否超出了矩阵的边界。若超出,则调整方向进行反射。
  2. 位置计算:

    • 使用模运算和反射逻辑来计算物体的位置。例如,x 轴的反射可以通过 (x + t * vx) % (2 * (cols - 1)) 计算,然后调整方向。
  3. 遍历每个时间单位:

    • 从时间 0 到时间 t,计算物体的位置并检查其值是否为 1,并统计经过 1 的次数。

Java 实现

import java.util.Scanner;

public class MatrixReflection {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        
        // 读取矩阵的行和列
        int rows = scanner.nextInt();
        int cols = scanner.nextInt();
        
        // 读取矩阵
        int[][] matrix = new int[rows][cols];
        for (int i = 0; i < rows; i++) {
            for (int j = 0; j < cols; j++) {
                matrix[i][j] = scanner.nextInt();
            }
        }
        
        // 读取初始位置和速度
        int startX = scanner.nextInt();
        int startY = scanner.nextInt();
        int vx = scanner.nextInt();
        int vy = scanner.nextInt();
        
        // 读取时间 t
        int t = scanner.nextInt();
        
        // 计算经过的1的点的次数
        int count = 0;
        
        for (int time = 0; time <= t; time++) {
            int x = startX + time * vx;
            int y = startY + time * vy;
            
            // 处理 x 方向反射
            int horizontalPeriod = 2 * (cols - 1);
            if (x < 0) {
                x = -x;
                x = (x / horizontalPeriod) % 2 == 0 ? x % horizontalPeriod : horizontalPeriod - x % horizontalPeriod;
            } else {
                x = x % horizontalPeriod;
                if (x >= cols) {
                    x = horizontalPeriod - x;
                }
            }
            
            // 处理 y 方向反射
            int verticalPeriod = 2 * (rows - 1);
            if (y < 0) {
                y = -y;
                y = (y / verticalPeriod) % 2 == 0 ? y % verticalPeriod : verticalPeriod - y % verticalPeriod;
            } else {
                y = y % verticalPeriod;
                if (y >= rows) {
                    y = verticalPeriod - y;
                }
            }
            
            // 检查当前位置是否是1
            if (matrix[y][x] == 1) {
                count++;
            }
        }
        
        // 输出结果
        System.out.println(count);
    }
}

说明

  1. 矩阵处理:

    • 读取矩阵并初始化。
  2. 位置计算:

    • 使用反射逻辑计算当前位置,并处理边界情况。
  3. 统计:

    • 遍历每个时间单位,检查当前位置并统计经过 1 的次数。


题目描述

给定一组人的能力值和一个比赛团队要求的最低能力值,每个团队可以由一人或两人组成,并且一个人只能参加一个团队。目标是计算出最多可以派出多少只符合要求的团队。

输入描述

  1. 总人数:一个整数 N,范围为 1500000
  2. 能力值数组:一个包含 N 个整数的数组,代表每个人的能力值。数组的大小和每个元素的值都在 1500000 之间。
  3. 最低能力值:一个整数 M,代表每个团队需要的最低能力值,范围为 1500000

输出描述

输出最多可以派出的团队数量。

示例

输入1

5
3 1 5 7 9
8

输出1

3

解释1

  • 能力值为 35 组成一个队伍。
  • 能力值为 17 组成一个队伍。
  • 能力值为 9 自己组成一个队伍。
  • 最终最多可以派出 3 个团队。

输入2

7
3 1 5 7 9 2 6
8

输出2

4

解释2

  • 能力值为 35 组成一个队伍。
  • 能力值为 17 组成一个队伍。
  • 能力值为 9 自己组成一个队伍。
  • 能力值为 26 组成一个队伍。
  • 最终最多可以派出 4 个团队。

输入3

3
1 1 9
8

输出3

1

解释3

  • 能力值为 9 自己组成一个队伍。
  • 剩余的能力值为 11 无法组成符合要求的队伍。
  • 最终最多可以派出 1 个团队。

解题思路

  1. 排序能力值:首先对能力值进行排序,以便后续更高效地找到配对。

  2. 使用双指针法

    • 一个指针从头开始遍历,另一个指针从尾部开始。
    • 尝试从两端找到符合要求的配对,并更新指针位置。
    • 如果两者的能力值之和满足最低要求,形成一个团队并移动指针。
    • 如果无法配对,尝试单独将当前最大能力值放入一个团队。
  3. 处理单独的能力值

    • 如果某些能力值无法与其他值配对,则单独成为一个团队。

Java 实现

import java.util.Arrays;
import java.util.Scanner;

public class TeamFormation {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // 读取总人数
        int n = scanner.nextInt();
        // 读取能力值数组
        int[] abilities = new int[n];
        for (int i = 0; i < n; i++) {
            abilities[i] = scanner.nextInt();
        }
        // 读取团队要求的最低能力值
        int minAbility = scanner.nextInt();

        // 处理逻辑
        System.out.println(maxTeams(abilities, minAbility));
    }

    public static int maxTeams(int[] abilities, int minAbility) {
        // 对能力值进行排序
        Arrays.sort(abilities);

        int left = 0;
        int right = abilities.length - 1;
        int teams = 0;

        // 使用双指针法
        while (left <= right) {
            // 如果当前最小值和最大值可以组成一个队伍
            if (abilities[left] + abilities[right] >= minAbility) {
                teams++;
                left++;
                right--;
            } else {
                // 否则,最小值不能和最大值配对,最小值单独处理
                left++;
            }
        }

        return teams;
    }
}

解释

  1. 排序:对能力值进行排序,使得我们可以从两端进行有效配对。
  2. 双指针法:使用两个指针来遍历数组,尝试找到符合条件的队伍。
  3. 更新指针:根据是否能配对来更新指针的位置。

该算法在排序的复杂度为 (O(n \log n)),遍历的复杂度为 (O(n)),因此整体复杂度是 (O(n \log n)),适合处理大规模数据。


题目描述

A 和 B 两个人玩抢7游戏。游戏规则如下:

  1. A 先报一个起始数字 (X)((10 \leq X \leq 10000))。
  2. B 报下一个数字 (Y)((X - Y < 3))。
  3. A 再报一个数字 (Z)((Y - Z < 3))。
  4. 游戏继续进行,直到某个人的报数是 7,该人即为胜者。

任务:计算在 B 赢得比赛的情况下,有多少种可能的数字组合。

输入描述

  • 起始数字 (M),其中 (10 \leq M \leq 10000)。

输出描述

  • B 能赢得比赛的组合次数。

示例

输入

10

输出

1

解释

  • 这个例子表示起始数字为 10,B 能赢得比赛的组合次数为 1。

解题思路

为了计算 B 赢得比赛的组合次数,我们需要模拟游戏的每一步直到游戏结束,并检查哪些情况下 B 能够赢得比赛。

步骤

  1. 递归模拟游戏:从起始数字开始,递归模拟 A 和 B 的每一步。检查所有符合规则的数字对,直到某人报数为 7。
  2. 终止条件:当某人报数为 7,检查是否是 B 赢得比赛。如果是 B 赢得比赛,记录下这种情况。
  3. 记忆化递归:为了避免重复计算相同状态,我们可以使用记忆化递归(缓存结果)来提高效率。

递归细节

  • 状态:当前的数字、当前轮到谁报数。
  • 递归函数:模拟每一步的选择,并递归进行下去。

Java 实现

下面是 Java 实现代码,通过递归和记忆化来解决问题:

import java.util.HashMap;
import java.util.Map;
import java.util.Scanner;

public class NumberGame {

    private static final int WINNING_NUMBER = 7;
    private static Map<String, Integer> memo = new HashMap<>();

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int startNumber = scanner.nextInt();
        System.out.println(countWinningCombinations(startNumber));
    }

    private static int countWinningCombinations(int startNumber) {
        return calculateWays(startNumber, true); // Start with A's turn
    }

    private static int calculateWays(int currentNumber, boolean isA) {
        if (currentNumber == WINNING_NUMBER) {
            return isA ? 0 : 1; // B wins if current number is 7 and it's B's turn
        }

        String key = currentNumber + "," + isA;
        if (memo.containsKey(key)) {
            return memo.get(key);
        }

        int ways = 0;
        for (int nextNumber = currentNumber - 2; nextNumber <= currentNumber + 2; nextNumber++) {
            if (nextNumber < 1) continue; // Ignore invalid numbers
            ways += calculateWays(nextNumber, !isA); // Switch turns
        }

        memo.put(key, ways);
        return ways;
    }
}

解释

  1. 递归函数 calculateWays:接受当前数字和一个布尔值 isA 来指示当前轮到谁报数。递归地计算每个可能的下一个数字,并根据是否 B 赢得比赛来更新计数。
  2. 记忆化:使用 memo 来存储计算过的状态结果,以避免重复计算。

此实现通过递归和记忆化来解决问题,适用于大范围的起始数字,并且能有效处理大量状态的计算。


题目描述

给定一个连续不包含空格的字符串,该字符串仅包含英文小写字母及英文标点符号(逗号、分号、句号),同时给定词库,对该字符串进行精确分词。

说明

  1. 精确分词:字符串分词后,不会出现重叠。即“ilovechina”不同词库可分割为“i,love,china”或“ilove,china”,不能分割出现重叠的“i,ilove,china”。
  2. 标点符号:不成词,仅用于断句。
  3. 词库:根据外部知识库统计出来的常用词汇。例如:dictionary=["i","love","china","lovechina","ilove"]
  4. 分词原则:采用分词顺序优先且最长匹配原则。

输入描述

  1. 第一行输入待分词语句,如"ilovechina"
  2. 第二行输入词库,如"i,love,china,ch,na,ve,lo,this,is,the,word"

输出描述
按顺序输出分词结果,如"i,love,china"

示例

示例1

输入:
ilovechina
i,love,china,ch,na,ve,lo,this,is,the,word

输出:
i,love,china

示例2

输入:
iat
i,love,china,ch,na,ve,lo,this,is,the,word,beauti,tiful,ful

输出:
i,a,t

示例3

输入:
ilovechina,thewordisbeautiful
i,love,china,ch,na,ve,lo,this,is,the,word,beauti,tiful,ful

输出:
i,love,china,the,word,is,beauti,ful

Java 代码

以下是 Java 实现代码:

import java.util.*;

public class ExactSegmentation {
    
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // Read the input string
        String inputString = scanner.nextLine().trim();

        // Read the dictionary
        String[] dictionary = scanner.nextLine().trim().split(",");

        // Call the segmentation method
        List<String> result = segmentString(inputString, dictionary);
        
        // Output the result
        System.out.println(String.join(",", result));
    }

    private static List<String> segmentString(String s, String[] dictionary) {
        Set<String> dictSet = new HashSet<>(Arrays.asList(dictionary));
        int n = s.length();
        
        // dp[i] will be the start index of the previous valid segment ending at i
        int[] dp = new int[n + 1];
        Arrays.fill(dp, -1);
        
        // To store the word that ends at index i
        String[] wordEnd = new String[n + 1];
        
        // Start filling dp and wordEnd
        for (int i = 0; i < n; i++) {
            for (int j = i; j < Math.min(n, i + 20); j++) {
                String substring = s.substring(i, j + 1);
                if (dictSet.contains(substring)) {
                    dp[j + 1] = i;
                    wordEnd[j + 1] = substring;
                }
            }
        }
        
        // Reconstruct the result
        List<String> result = new ArrayList<>();
        int idx = n;
        while (idx > 0) {
            if (dp[idx] == -1) {
                // This should not happen as per problem statement
                result.add(String.valueOf(s.charAt(idx - 1)));
                idx--;
            } else {
                result.add(wordEnd[idx]);
                idx = dp[idx];
            }
        }
        
        Collections.reverse(result);
        return result;
    }
}

说明

  1. 输入读取

    • 从控制台读取待分词的字符串和词库。
  2. 动态规划

    • 使用 dp 数组记录每个位置的前一个有效分割位置。
    • 使用 wordEnd 数组记录每个位置的词汇。
  3. 分词计算

    • 使用滑动窗口方法查找最长的匹配词,并更新 dpwordEnd
  4. 结果输出

    • 将分词结果按顺序输出,以逗号分隔。


题目描述

给定一个由 1n 的连续数字组成的数组,其中 n 为 3 的倍数。每次从数组中取出 3 个元素,去掉这 3 个元素中的一个最大值和一个最小值,并将剩下的元素的值累加到 SS 的初始值为 0。

可以通过调整数组中元素的位置来改变最终的结果,每移动一个元素计为一次移动。请计算最少移动次数以使得数组和 S 最大。

输入描述

  • 数组长度 n 的范围为 [3, 600]
  • 数组中数字的范围为 [1, 10000]
  • 数组由一个字符串表示,不同数字元素之间使用空格分隔

输出描述

  • 移动次数是一个自然数
  • 如果无需移动,返回0

示例

示例1

输入:
1 8 9 7 4 2 5 6 3

输出:
1

解释
通过将 1 移动到末尾,可以使得每次取出的 3 个元素中,保留的元素尽量大,从而使得 S 最大。最少需要 1 次移动。

Java 代码

以下是 Java 实现代码:

import java.util.*;

public class MaxSumWithMinMoves {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        String[] input = scanner.nextLine().split(" ");
        int[] array = Arrays.stream(input).mapToInt(Integer::parseInt).toArray();
        
        int n = array.length;
        int[] sortedArray = array.clone();
        Arrays.sort(sortedArray);

        // Calculate the ideal sum and how many movements are needed
        int idealSum = 0;
        for (int i = 0; i < n / 3; i++) {
            idealSum += sortedArray[3 * i + 1];
        }

        // Calculate the actual sum
        int actualSum = 0;
        int[] positions = new int[n];
        for (int i = 0; i < n; i++) {
            positions[array[i] - 1] = i;
        }

        for (int i = 0; i < n / 3; i++) {
            int[] triplet = new int[3];
            for (int j = 0; j < 3; j++) {
                triplet[j] = array[i * 3 + j];
            }
            Arrays.sort(triplet);
            actualSum += triplet[1];
        }

        // Calculate the number of moves needed
        int moves = 0;
        for (int i = 0; i < n; i++) {
            if (array[i] != sortedArray[i]) {
                moves++;
            }
        }
        
        // The moves count is divided by 2 as each swap adjusts two positions
        System.out.println(moves / 2);
    }
}

说明

  1. 输入读取

    • 从标准输入读取数组,按空格分隔。
  2. 理想和计算

    • 对数组进行排序,计算理想的最大 S 和。
  3. 实际和计算

    • 计算原始数组中每组三个元素保留中间值的和。
  4. 最少移动计算

    • 通过比较排序后的数组与原始数组,计算最少需要的移动次数。每次交换两个元素的位置,所以总移动次数需要除以 2。
  5. 结果输出

    • 输出最少的移动次数以使得 S 最大。


题目描述

给定一个二维数组,包含 MN 列,每个元素代表一个像素,像素值为 1 或 5。你需要计算所有像素值为 1 的物体的边界个数,其中边界定义为与像素值为 5 相邻的像素值为 1 的区域。两个像素值为 1 的区域相邻且是边界的区域被认为是同一个边界。

输入描述

  • 第一行:两个整数 M 和 N,表示二维数组的行数和列数。
  • 接下来的 M 行:每行包含 N 个数字,表示二维数组的像素值,仅包含 1 和 5。

输出描述

  • 输出像素值为 1 的物体的边界个数。如果没有边界,输出 0。

示例

示例 1

输入:
3 3
1 1 1
1 1 1
5 5 5

输出:
1

解释
像素值为 1 的物体(全为1的区域)与像素值为 5 的物体相邻,因此形成了一个边界。

示例 2

输入:
6 6
1 1 1 5 5 5
1 1 1 5 5 5
1 1 1 5 5 5
5 5 5 1 1 1
5 5 5 1 1 1
5 5 5 1 1 1

输出:
2

解释
有两个不同的像素值为 1 的边界区域,一个在左上角,与像素值为 5 的物体相邻,另一个在右下角,同样与像素值为 5 的物体相邻。

Java 代码

以下是 Java 实现代码,用于计算像素值为 1 的边界数量:

import java.util.Scanner;
import java.util.LinkedList;
import java.util.Queue;

public class BoundaryCount {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int M = scanner.nextInt();
        int N = scanner.nextInt();
        scanner.nextLine();  // Consume the newline character

        int[][] grid = new int[M][N];
        for (int i = 0; i < M; i++) {
            for (int j = 0; j < N; j++) {
                grid[i][j] = scanner.nextInt();
            }
            scanner.nextLine();  // Consume the newline character
        }

        System.out.println(countBoundaries(grid, M, N));
    }

    private static int countBoundaries(int[][] grid, int M, int N) {
        boolean[][] visited = new boolean[M][N];
        int boundaryCount = 0;

        for (int i = 0; i < M; i++) {
            for (int j = 0; j < N; j++) {
                if (grid[i][j] == 1 && !visited[i][j]) {
                    if (isBoundary(grid, M, N, i, j, visited)) {
                        boundaryCount++;
                    }
                }
            }
        }
        return boundaryCount;
    }

    private static boolean isBoundary(int[][] grid, int M, int N, int i, int j, boolean[][] visited) {
        boolean isBoundary = false;
        Queue<int[]> queue = new LinkedList<>();
        queue.add(new int[]{i, j});
        visited[i][j] = true;

        // 8 directions: up, down, left, right, and 4 diagonals
        int[] directions = {-1, 0, 1, 0, -1, -1, -1, 1, 1, -1, 1, 1, -1, 1, 1, -1};
        
        while (!queue.isEmpty()) {
            int[] current = queue.poll();
            int x = current[0];
            int y = current[1];

            for (int d = 0; d < directions.length; d += 2) {
                int newX = x + directions[d];
                int newY = y + directions[d + 1];

                if (newX >= 0 && newX < M && newY >= 0 && newY < N) {
                    if (grid[newX][newY] == 1) {
                        if (!visited[newX][newY]) {
                            visited[newX][newY] = true;
                            queue.add(new int[]{newX, newY});
                        }
                    } else if (grid[newX][newY] == 5) {
                        isBoundary = true;
                    }
                }
            }
        }
        return isBoundary;
    }
}

代码说明

  1. 输入读取:从标准输入读取数组的维度和像素值,构造二维数组 grid

  2. 边界计算

    • 使用 BFS 遍历像素值为 1 的区域,检查是否与像素值为 5 的区域相邻。
    • 使用 visited 数组避免重复访问。
  3. 方向数组

    • 用于检查当前像素的 8 个方向,包括水平、垂直和对角线方向。
  4. 输出结果

    • 输出像素值为 1 的边界区域的个数。


题目描述

给定一个矩阵和一个包含特定整数的数组,要求找出矩阵中最小宽度的子矩阵,这个子矩阵包含数组中的所有整数。如果找不到符合要求的子矩阵,则返回 -1。

输入描述

  1. 第一行包含两个整数 NM,分别表示矩阵的行数和列数。
  2. 接下来 N 行,每行包含 M 个整数,表示矩阵的内容。
  3. 下一行包含一个正整数 K,表示所需包含的数组的大小。
  4. 第四行包含 K 个整数,表示所需包含的数组,数组中可能有重复的数字。

输出描述

  • 输出一个整数,表示满足要求子矩阵的最小宽度。如果找不到符合要求的子矩阵,输出 -1。

示例

示例 1

输入:
2 5
1 2 2 3 1
2 3 2 3 2
3
1 2 3

输出:
2

解释
矩阵的第0列到第3列包含了所有的 1, 2, 3。其中最小的宽度是2,即第0列到第1列。

示例 2

输入:
2 5
1 2 2 3 1
1 3 2 3 4
3
1 1 4

输出:
5

解释
矩阵的第1列到第5列包含了所有的 1, 1, 4。其中最小的宽度是5,即第1列到第5列。

Java 代码

以下是解决该问题的 Java 代码:

import java.util.*;

public class MinWidthSubmatrix {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        
        // Read matrix dimensions
        int N = scanner.nextInt();
        int M = scanner.nextInt();
        scanner.nextLine(); // Consume the newline character

        // Read matrix
        int[][] matrix = new int[N][M];
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < M; j++) {
                matrix[i][j] = scanner.nextInt();
            }
            scanner.nextLine(); // Consume the newline character
        }

        // Read K
        int K = scanner.nextInt();
        scanner.nextLine(); // Consume the newline character

        // Read required numbers
        int[] required = new int[K];
        for (int i = 0; i < K; i++) {
            required[i] = scanner.nextInt();
        }

        // Find the minimum width of submatrix
        System.out.println(findMinWidth(matrix, N, M, required));
    }

    private static int findMinWidth(int[][] matrix, int N, int M, int[] required) {
        // Count occurrences of each required number
        Map<Integer, Integer> countMap = new HashMap<>();
        for (int num : required) {
            countMap.put(num, countMap.getOrDefault(num, 0) + 1);
        }

        int minWidth = Integer.MAX_VALUE;

        // Check all possible submatrices
        for (int left = 0; left < M; left++) {
            // Map to keep track of number counts in the current window
            Map<Integer, Integer> currentCount = new HashMap<>();
            int right = left;

            while (right < M) {
                // Update the count map with the new column
                for (int i = 0; i < N; i++) {
                    int value = matrix[i][right];
                    currentCount.put(value, currentCount.getOrDefault(value, 0) + 1);
                }

                // Check if the current window satisfies the condition
                if (isValid(currentCount, countMap)) {
                    minWidth = Math.min(minWidth, right - left + 1);
                }

                right++;
            }
        }

        return minWidth == Integer.MAX_VALUE ? -1 : minWidth;
    }

    private static boolean isValid(Map<Integer, Integer> currentCount, Map<Integer, Integer> requiredCount) {
        for (Map.Entry<Integer, Integer> entry : requiredCount.entrySet()) {
            int num = entry.getKey();
            int required = entry.getValue();
            if (currentCount.getOrDefault(num, 0) < required) {
                return false;
            }
        }
        return true;
    }
}

代码说明

  1. 输入读取:读取矩阵的尺寸、矩阵内容、目标数组的大小和目标数组。

  2. 初始化:使用 HashMap 记录目标数组中每个数字的频次。

  3. 查找最小宽度

    • 遍历所有可能的子矩阵的左边界 left 和右边界 right
    • 对于每一个可能的 leftright,计算当前子矩阵中每个数字的频次,并更新 currentCount
    • 使用 isValid 方法检查当前子矩阵是否满足包含所有目标数字的条件。
  4. 输出结果:输出最小宽度,如果找不到满足条件的子矩阵,则输出 -1。



题目描述

从一个 (N \times M) 的矩阵中选出 (N) 个数,要求每个选出的数字都来自不同的行和不同的列。你的任务是找出选出的 (N) 个数中第 (K) 大的数字的最小值。

输入描述

  • 第一行包含三个整数 (N), (M), (K),分别表示矩阵的行数、列数和所需的第 (K) 大的数字。
  • 接下来 (N) 行每行包含 (M) 个整数,表示矩阵的内容。

输出描述

  • 输出一个整数,表示所求第 (K) 大的数字的最小值。

示例

输入

3 4 2
1 5 6 6
8 3 4 3
6 8 6 3

输出

3

解释
从矩阵中可以选出以下几种组合,计算每种组合中第 (2) 大的数字:

  • 选择 1, 3, 6 -> 第 2 大的数字是 3
  • 选择 1, 3, 3 -> 第 2 大的数字是 3
  • 选择 1, 4, 8 -> 第 2 大的数字是 4
  • 选择 1, 4, 3 -> 第 2 大的数字是 4
  • ...

最终,第 (2) 大的数字的最小值是 3。

解题思路

  1. 生成所有可能的选数组合

    • 使用回溯算法或其它方法生成所有可能的选数组合,确保每个选出的数字都来自不同的行和列。
  2. 计算每个组合中第 (K) 大的数字

    • 对于每一个组合,取出 (N) 个数字,找到其中的第 (K) 大的数字。
  3. 找到第 (K) 大的数字的最小值

    • 在所有可能的组合中,找到第 (K) 大的数字的最小值。

Java 实现

以下是 Java 实现的代码:

import java.util.*;

public class MatrixSelection {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // Read input
        int N = scanner.nextInt();
        int M = scanner.nextInt();
        int K = scanner.nextInt();
        scanner.nextLine(); // consume newline

        int[][] matrix = new int[N][M];
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < M; j++) {
                matrix[i][j] = scanner.nextInt();
            }
            scanner.nextLine(); // consume newline
        }

        // Find minimum value of K-th largest number
        System.out.println(findMinKthLargest(matrix, N, M, K));
    }

    private static int findMinKthLargest(int[][] matrix, int N, int M, int K) {
        List<Integer> allPossibleValues = new ArrayList<>();
        List<int[]> combinations = generateCombinations(N, M);

        for (int[] comb : combinations) {
            List<Integer> values = new ArrayList<>();
            for (int i = 0; i < N; i++) {
                values.add(matrix[i][comb[i]]);
            }
            Collections.sort(values, Collections.reverseOrder());
            if (values.size() >= K) {
                allPossibleValues.add(values.get(K - 1));
            }
        }

        if (allPossibleValues.isEmpty()) {
            return -1; // if no valid combination is found
        }

        Collections.sort(allPossibleValues);
        return allPossibleValues.get(0);
    }

    private static List<int[]> generateCombinations(int N, int M) {
        List<int[]> result = new ArrayList<>();
        int[] comb = new int[N];
        generateCombinationsUtil(result, comb, 0, M, 0);
        return result;
    }

    private static void generateCombinationsUtil(List<int[]> result, int[] comb, int start, int M, int depth) {
        if (depth == comb.length) {
            result.add(comb.clone());
            return;
        }

        for (int i = start; i < M; i++) {
            comb[depth] = i;
            generateCombinationsUtil(result, comb, i + 1, M, depth + 1);
        }
    }
}

代码说明

  1. 输入读取:读取矩阵的尺寸、矩阵内容和所需的第 (K) 大的数字。
  2. 生成所有可能的列组合:使用递归生成所有可能的列组合。
  3. 计算每个组合的第 (K) 大数字
    • 对于每个组合,取出相应的矩阵元素,计算第 (K) 大的数字。
  4. 找到最小值
    • 找到所有组合中第 (K) 大的数字的最小值,并输出。


问题描述
为了充分发挥GPU算力,需要尽可能多的将任务交给GPU执行,现在有一个任务数组,数组元素表示在这1秒内新增的任务个数且每秒都有新增任务。

假设GPU最多一次执行n个任务,一次执行耗时1秒,在保证GPU不空闲情况下,最少需要多长时间执行完成。

输入描述:

第一个参数为GPU一次最多执行的任务个数,取值范围[1, 10000]
第二个参数为任务数组长度,取值范围[1, 10000]
第三个参数为任务数组,数字范围[1, 10000]
输出描述:

执行完所有任务最少需要多少秒。
示例1
输入
3
5
1 2 3 4 5
输出
6
说明
一次最多执行3个任务,最少耗时6s

示例2
输入
4
5
5 4 1 1 1
输出
5

为了解决这个问题,我们需要计算一个最小的时间 T,使得在给定的任务数组中,每秒可以安排一次执行至多 n 个任务,并且任务不会因为 GPU 处于空闲状态而拖延。

思路

  1. 任务数组和 GPU 执行能力:

    • 每秒 GPU 可以最多处理 n 个任务。
    • 给定一个任务数组 tasks,每个元素表示在那一秒内到达的任务数量。
  2. 计算最少需要多少秒:

    • 需要找出一个最小的 T,使得 T 秒内 GPU 可以处理所有任务,并且 GPU 不会处于空闲状态。
  3. 二分查找:

    • 我们可以使用二分查找来寻找这个最小的 T。假设最大可能的 T 为任务数组的长度加上一个安全余量(比如最大单个任务数)。
    • 对于每个可能的 T,模拟 T 秒内的任务处理情况,检查是否可以完成所有任务。

详细步骤

  1. 计算是否能在给定时间内完成所有任务:

    • 对于一个给定的 T,我们可以在每秒最多执行 n 个任务。计算每秒可以处理的任务数,并将其与实际需要处理的任务数进行比较。
    • 如果在 T 秒内可以完成所有任务,则 T 是一个合适的时间。
  2. 二分查找的实现:

    • 设定 low 为1,high 为任务数组长度的最大可能值。
    • 对于每个中间值 mid,模拟 mid 秒内的任务处理情况,更新 lowhigh 直到找到最小的满足条件的 T

Java 实现

import java.util.Scanner;

public class MinExecutionTime {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // 读取输入
        int maxTasksPerSecond = scanner.nextInt();
        int numSeconds = scanner.nextInt();
        int[] tasks = new int[numSeconds];
        
        for (int i = 0; i < numSeconds; i++) {
            tasks[i] = scanner.nextInt();
        }

        // 计算最小执行时间
        System.out.println(findMinExecutionTime(maxTasksPerSecond, tasks));
    }

    private static int findMinExecutionTime(int n, int[] tasks) {
        int left = 1;
        int right = tasks.length;
        
        // 二分查找
        while (left < right) {
            int mid = left + (right - left) / 2;
            if (canCompleteInTime(n, tasks, mid)) {
                right = mid;
            } else {
                left = mid + 1;
            }
        }
        return left;
    }

    private static boolean canCompleteInTime(int n, int[] tasks, int time) {
        int[] remainingTasks = new int[time];
        // 将任务数组复制到对应的时间数组中
        for (int i = 0; i < tasks.length; i++) {
            int end = Math.min(i + time, time);
            for (int j = i; j < end; j++) {
                remainingTasks[j - i] += tasks[i];
            }
        }
        // 检查是否可以在给定时间内完成
        for (int i = 0; i < time; i++) {
            if (remainingTasks[i] > n) {
                return false;
            }
        }
        return true;
    }
}

代码解释

  1. 读取输入:

    • 从输入中读取 GPU 的最大任务数、任务数组长度以及任务数组。
  2. 二分查找:

    • 使用二分查找来寻找最小的时间 T,使得在 T 秒内可以完成所有任务。
  3. 时间验证:

    • 对于每一个可能的时间 T,计算是否能在 T 秒内完成所有任务,并根据结果调整二分查找的范围。
  4. 输出结果:

    • 输出最小的时间 T


某业务需要根据终端的IP地址获取该终端归属的城市,可以根据公开的IP地址池信息查询归属城市。地址池格式如下城市名=起始IP,结束IP,起始和结束地址按照英文逗号分隔,多个地址段采用英文分号分隔。

比如City1=1.1.1.1,1.1.1.2;City1=1.1.1.11,1.1.1.16;City2=3.3.3.3,4.4.4.4;City3=2.2.2.2,6.6.6.6。

一个城市可以有多个IP段,比如City1有2个IP段。城市间也可能存在包含关系,如City3的IP段范围包含City2的IP段范围。现在要根据输入的IP列表,返回最佳匹配的城市列表。

注:最佳匹配即包含待查询IP且长度最小的IP段,比如例子中3.4.4.4最佳匹配是City2=3.3.3.3,4.4.4.4,5.5.5.5的最佳匹配是City3=2.2.2.2,6.6.6.6。

输入描述:

输入共2行。

第一行为城市的IP段列表,多个IP段采用英文分号';'分隔,IP段列表最大不超过500000。城市名称只包含英文字母、数字和下划线,最多不超过100000个。IP段包含关系可能有多层,但不超过100层。

第二行为查询的IP列表,多个IP采用英文逗号‘,’分隔,最多不超过10000条。

输出描述:

最佳匹配的城市名列表,采用英文逗号‘,’分隔,城市列表长度应该跟查询的IP列表长度一致。

为了实现根据终端 IP 地址获取归属城市的功能,我们需要解析 IP 地址池,并对每个查询的 IP 地址进行匹配,找到最精确的城市归属。以下是详细的步骤和代码实现:

步骤解析

  1. 解析城市 IP 地址池:

    • 读取城市和 IP 段的映射关系。
    • 将每个城市的 IP 段解析成起始 IP 和结束 IP。
    • 将这些 IP 段存储在适当的数据结构中,方便后续查询。
  2. IP 地址转换:

    • 将 IP 地址从字符串格式转换成整数格式,便于比较和计算。IP 地址的转换方法是:a.b.c.d 转换成 a * 256^3 + b * 256^2 + c * 256 + d
  3. 匹配查询 IP:

    • 对于每个查询 IP 地址,将其转换成整数,并查找其所在的城市。
    • 检查 IP 是否落在城市的 IP 段范围内,找到最精确的匹配。
  4. 优化:

    • 使用高效的数据结构如 TreeMapList 进行 IP 地址段的存储和查询。

代码实现

以下是使用 Java 语言实现的解决方案:

import java.util.*;
import java.util.regex.Pattern;

public class IPAddressMatcher {

    private static final String IP_PATTERN = "\\d+\\.\\d+\\.\\d+\\.\\d+";

    static class IPRange {
        String city;
        long start;
        long end;

        IPRange(String city, long start, long end) {
            this.city = city;
            this.start = start;
            this.end = end;
        }

        boolean contains(long ip) {
            return ip >= start && ip <= end;
        }
    }

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        // 读取城市的IP段列表
        String cityList = scanner.nextLine();
        // 读取查询的IP列表
        String ipList = scanner.nextLine();

        // 解析城市 IP 段
        List<IPRange> ipRanges = parseCityIPRanges(cityList);

        // 解析查询 IP 列表
        List<String> queries = Arrays.asList(ipList.split(","));

        // 查找每个查询 IP 的最佳匹配城市
        List<String> results = new ArrayList<>();
        for (String ip : queries) {
            long ipInt = ipToLong(ip);
            String city = findBestMatch(ipRanges, ipInt);
            results.add(city);
        }

        // 输出结果
        System.out.println(String.join(",", results));
    }

    private static List<IPRange> parseCityIPRanges(String cityList) {
        List<IPRange> ranges = new ArrayList<>();
        String[] citySegments = cityList.split(";");
        for (String segment : citySegments) {
            String[] parts = segment.split("=");
            if (parts.length == 2) {
                String city = parts[0].trim();
                String[] ipRanges = parts[1].split(",");
                for (int i = 0; i < ipRanges.length; i += 2) {
                    long start = ipToLong(ipRanges[i]);
                    long end = ipToLong(ipRanges[i + 1]);
                    ranges.add(new IPRange(city, start, end));
                }
            }
        }
        return ranges;
    }

    private static long ipToLong(String ip) {
        if (!Pattern.matches(IP_PATTERN, ip)) {
            throw new IllegalArgumentException("Invalid IP address format");
        }
        String[] parts = ip.split("\\.");
        return (Long.parseLong(parts[0]) << 24) |
               (Long.parseLong(parts[1]) << 16) |
               (Long.parseLong(parts[2]) << 8) |
               Long.parseLong(parts[3]);
    }

    private static String findBestMatch(List<IPRange> ranges, long ip) {
        String bestCity = "";
        long bestEnd = Long.MAX_VALUE;
        for (IPRange range : ranges) {
            if (range.contains(ip)) {
                if (range.end < bestEnd) {
                    bestEnd = range.end;
                    bestCity = range.city;
                }
            }
        }
        return bestCity;
    }
}

代码解释

  1. IP 地址转换:

    • 使用 ipToLong 方法将 IP 地址转换为一个长整型值,便于范围比较。
  2. 解析城市 IP 段:

    • parseCityIPRanges 方法解析城市的 IP 段列表,创建 IP 范围对象 IPRange 存储城市及其 IP 范围。
  3. 查找最佳匹配:

    • findBestMatch 方法根据 IP 地址在 IP 范围中查找最精确的匹配城市。我们通过检查 IP 是否在范围内,并选择结束地址最小的城市。
  4. 输入和输出:

    • Scanner 中读取输入数据,并按照格式解析和处理,最后输出结果。

这个方法确保了在给定的输入数据范围内能够高效地找到每个查询 IP 的最佳匹配城市。



题目描述

n 个学生排成一排,学生编号分别是 1 到 n,其中 n 是 3 的整倍数。老师通过随机抽签将所有学生分成 m 个 3 人小组(n = 3m)。为了便于同组学生交流,老师决定将小组成员安排到一起,即同组成员彼此相连,同组任意两个成员之间不能有其它组的成员。老师可以调整队伍,每次可以将任何一名学生移动到队伍的任意位置,这被计为一次调整。请计算最少需要多少次调整才能达到目标。注意:

  • 小组之间的顺序没有要求。
  • 同组学生之间的顺序没有要求。

输入描述:

输入包含两行,每行为空格分隔的整数,表示学生编号。

  • 第一行表示学生目前的排队情况
  • 第二行表示随机抽签的分组情况,从左开始每 3 个元素为一组

n 为学生的数量,范围为 [3, 900],且 n 一定为 3 的整数倍。两行的元素个数相同。

输出描述:

输出一个整数,表示老师调整学生达到同组彼此相连的最少调整次数。

解题思路

  1. 解析输入数据:

    • 读取当前学生的排列顺序和目标分组情况。
    • 确定每组的目标位置范围。
  2. 计算每组的目标位置:

    • 确定每组在目标位置上学生的索引。
    • 将当前组内学生与目标组的排序进行对比,计算需要的最少调整次数。
  3. 最小调整次数计算:

    • 对每组学生,找到当前学生的位置,并计算与目标位置的匹配程度,进而计算出所需的调整次数。

Java 实现

下面是 Java 代码实现:

import java.util.*;

public class Main {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        
        // Read the input data
        String[] currentLine = scanner.nextLine().split(" ");
        String[] targetLine = scanner.nextLine().split(" ");
        
        int n = currentLine.length;
        int m = n / 3;
        
        // Parse current students and target groups
        int[] currentStudents = Arrays.stream(currentLine).mapToInt(Integer::parseInt).toArray();
        int[][] targetGroups = new int[m][3];
        for (int i = 0; i < m; i++) {
            targetGroups[i] = Arrays.copyOfRange(Arrays.stream(targetLine).mapToInt(Integer::parseInt).toArray(), 3 * i, 3 * i + 3);
        }
        
        // To store the minimum moves required
        int minMoves = 0;
        
        // Create a map for quick lookup
        Map<Integer, List<Integer>> studentGroupMap = new HashMap<>();
        for (int i = 0; i < n; i++) {
            studentGroupMap.computeIfAbsent(currentStudents[i], k -> new ArrayList<>()).add(i);
        }
        
        // Iterate through each group
        for (int i = 0; i < m; i++) {
            Set<Integer> groupSet = new HashSet<>(Arrays.asList(targetGroups[i][0], targetGroups[i][1], targetGroups[i][2]));
            List<Integer> groupIndices = new ArrayList<>();
            for (int student : groupSet) {
                groupIndices.addAll(studentGroupMap.get(student));
            }
            groupIndices.sort(Comparator.naturalOrder());
            
            // Determine target positions for the group
            List<Integer> targetIndices = new ArrayList<>();
            for (int j = 0; j < 3; j++) {
                targetIndices.add(3 * i + j);
            }
            
            // Compute minimum moves required to match target indices
            minMoves += computeMinMoves(groupIndices, targetIndices);
        }
        
        // Output the result
        System.out.println(minMoves);
    }

    private static int computeMinMoves(List<Integer> currentIndices, List<Integer> targetIndices) {
        int moves = 0;
        for (int i = 0; i < 3; i++) {
            if (!currentIndices.get(i).equals(targetIndices.get(i))) {
                moves++;
            }
        }
        return moves / 2; // Each swap corrects two positions
    }
}

代码解释

  1. 输入解析:

    • 读取当前排列和目标分组数据,并转换为整型数组。
  2. 目标位置确定:

    • 对每组的目标位置进行排序,计算与当前学生位置的匹配度。
  3. 最小调整次数计算:

    • 计算每组需要的最少调整次数,并累计到总的调整次数中。
  4. 输出结果:

    • 打印最少的调整次数。

这个 Java 代码可以用于处理最大 900 名学生的情况,并计算将学生调整到目标位置的最少次数。

posted @ 2024-08-11 09:36  MingYu15  阅读(28)  评论(0)    收藏  举报