0

0

计算将数组转换为目标数组所需的最少分组数

碧海醫心

碧海醫心

发布时间:2025-11-22 17:45:02

|

228人浏览过

|

来源于php中文网

原创

计算将数组转换为目标数组所需的最少分组数

本文介绍了一种高效算法,用于确定将一个给定数组通过切割成最少连续片段并重新排列,以转换为另一个目标数组所需的最少分组数量。核心思想是利用目标数组的元素索引映射,遍历原始数组,通过比较元素在目标数组中的相对位置来识别连续的有序片段,从而计算出必要的分组数。

在处理数组转换问题时,我们有时需要将一个数组切割成若干连续的子数组,然后通过重新排列这些子数组来形成另一个目标数组。我们的目标是找出实现这一转换所需的最少子数组(或称分组)数量。本文将详细阐述一种基于索引映射的解决方案,该方案在给定数组元素唯一且长度相同的情况下表现高效。

问题分析

假设我们有两个数组 arr1 和 arr2,它们包含相同的唯一元素,只是顺序不同。例如,arr1 = [1, 4, 3, 2] 和 arr2 = [1, 2, 4, 3]。我们需要将 arr1 切割成最少数量的连续片段,然后重新排列这些片段以得到 arr2。

例如,对于 arr1 = [1, 4, 3, 2] 和 arr2 = [1, 2, 4, 3]: 我们可以将 arr1 切割为 (1), (4, 3), (2) 三个片段。 然后重新排列为 (1), (2), (4, 3) 即可得到 arr2。因此,答案是 3 个片段。

一个常见的误区是简单地计算两个数组中不同位置元素的数量。这种方法无法捕捉到片段重排的本质,因为即使元素位置不同,它们仍可能属于同一个可移动的连续片段。正确的思路是识别 arr1 中哪些元素序列在 arr2 中保持了相对的连续性。

核心算法思想

由于数组中的所有元素都是唯一的,我们可以利用这一特性。算法的核心思想是:

Spell.tools
Spell.tools

高颜值AI内容营销创作工具

下载
  1. 首先,创建一个映射(Map),将目标数组 arr2 中的每个元素与其在 arr2 中的索引关联起来。这将帮助我们快速查找 arr1 中元素在 arr2 中的期望位置。
  2. 然后,遍历 arr1。我们维护一个计数器 groupCount 来记录所需的分组数,并维护一个 prevIndexInTarget 变量,表示 arr1 中当前处理的元素在 arr2 中的预期索引。
  3. 对于 arr1 中的每个元素(从第二个元素开始),查找其在 arr2 中的索引 currentIndexInTarget。
  4. 如果 currentIndexInTarget 等于 prevIndexInTarget + 1,这意味着当前元素紧接着前一个元素在 arr2 中出现,它们可以构成一个连续的片段。此时,我们只需更新 prevIndexInTarget 为 currentIndexInTarget,并继续将它们视为同一个分组的一部分。
  5. 如果 currentIndexInTarget 不等于 prevIndexInTarget + 1,这意味着当前元素在 arr2 中的位置与前一个元素不连续,因此它必须开启一个新的分组。此时,我们需要增加 groupCount,并将 prevIndexInTarget 更新为 currentIndexInTarget。

初始时,第一个元素总是开启一个新的分组,所以 groupCount 初始化为 1。

详细实现步骤

  1. 构建索引映射: 遍历目标数组 arr2,将每个元素作为键,其在数组中的索引作为值,存入 Map 中。
  2. 初始化:
    • groupCount = 1:至少需要一个分组。
    • prevIndexInTarget = indexByValue.get(arr1[0]):获取 arr1 第一个元素在 arr2 中的索引。
  3. 遍历 arr1: 从 arr1 的第二个元素开始,迭代到数组末尾。
    • 在每次迭代中,获取当前元素 arr1[i] 在 arr2 中的索引 currentIndexInTarget = indexByValue.get(arr1[i])。
    • 判断连续性:
      • 如果 currentIndexInTarget == prevIndexInTarget + 1,则表示当前元素与前一个元素在 arr2 中是连续的,属于同一分组。更新 prevIndexInTarget = currentIndexInTarget。
      • 否则(currentIndexInTarget != prevIndexInTarget + 1),表示当前元素打破了连续性,需要开启一个新的分组。增加 groupCount++,并更新 prevIndexInTarget = currentIndexInTarget。
  4. 返回结果: 循环结束后,groupCount 即为所需的最少分组数。

示例代码 (Java)

以下是该算法的 Java 实现:

import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.IntStream;

public class ArrayGroupingConverter {

    /**
     * 计算将 arr1 转换为 arr2 所需的最少分组数。
     * 
     * @param arr1 原始数组
     * @param arr2 目标数组
     * @return 最少分组数
     */
    public static int calculateMinGroups(int[] arr1, int[] arr2) {
        // 1. 构建 arr2 的元素到索引的映射
        Map indexByValue = mapIndices(arr2);

        // 初始化分组计数器为 1 (至少一个分组)
        int groupCount = 1;

        // 获取 arr1 第一个元素在 arr2 中的索引,作为当前分组的起始索引
        int prevIndexInTarget = indexByValue.get(arr1[0]);

        // 2. 遍历 arr1,从第二个元素开始
        for (int i = 1; i < arr1.length; i++) {
            // 获取当前 arr1 元素在 arr2 中的索引
            int currentIndexInTarget = indexByValue.get(arr1[i]);

            // 3. 判断当前元素与前一个元素在 arr2 中是否连续
            if (currentIndexInTarget == prevIndexInTarget + 1) {
                // 如果连续,则它们属于同一个分组,更新前一个索引
                prevIndexInTarget++;
            } else {
                // 如果不连续,则需要开启一个新的分组
                groupCount++;
                // 更新前一个索引为当前元素的索引,作为新分组的起始
                prevIndexInTarget = currentIndexInTarget;
            }
        }

        return groupCount;
    }

    /**
     * 辅助方法:将数组元素映射到其索引。
     * 
     * @param arr 要映射的数组
     * @return 元素到索引的映射
     */
    public static Map mapIndices(int[] arr) {
        return IntStream.range(0, arr.length)
            .boxed()
            .collect(Collectors.toMap(
                i -> arr[i], // 键:数组元素
                Function.identity() // 值:元素索引
            ));
    }

    public static void main(String[] args) {
        int[] arr1 = {1, 4, 3, 2};
        int[] arr2 = {1, 2, 4, 3};
        System.out.println("原始数组: " + java.util.Arrays.toString(arr1));
        System.out.println("目标数组: " + java.util.Arrays.toString(arr2));
        System.out.println("所需的最少分组数: " + calculateMinGroups(arr1, arr2)); // 预期输出: 3

        int[] arr3 = {1, 2, 3, 4};
        int[] arr4 = {1, 2, 3, 4};
        System.out.println("\n原始数组: " + java.util.Arrays.toString(arr3));
        System.out.println("目标数组: " + java.util.Arrays.toString(arr4));
        System.out.println("所需的最少分组数: " + calculateMinGroups(arr3, arr4)); // 预期输出: 1

        int[] arr5 = {4, 3, 2, 1};
        int[] arr6 = {1, 2, 3, 4};
        System.out.println("\n原始数组: " + java.util.Arrays.toString(arr5));
        System.out.println("目标数组: " + java.util.Arrays.toString(arr6));
        System.out.println("所需的最少分组数: " + calculateMinGroups(arr5, arr6)); // 预期输出: 4
    }
}

输出结果:

原始数组: [1, 4, 3, 2]
目标数组: [1, 2, 4, 3]
所需的最少分组数: 3

原始数组: [1, 2, 3, 4]
目标数组: [1, 2, 3, 4]
所需的最少分组数: 1

原始数组: [4, 3, 2, 1]
目标数组: [1, 2, 3, 4]
所需的最少分组数: 4

注意事项与性能分析

  • 唯一性约束: 该算法的关键在于元素在数组中的唯一性。如果存在重复元素,则需要更复杂的逻辑来处理,因为单个元素可能对应多个目标索引

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

833

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

738

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

734

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

2

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号