[路飞]_LeetCode_1122. 数组的相对排序

197 阅读2分钟

「这是我参与2022首次更文挑战的第40天,活动详情查看:2022首次更文挑战

题目

给你两个数组,arr1 和 arr2,arr2 中的元素各不相同,arr2 中的每个元素都出现在 arr1 中。

对 arr1 中的元素进行排序,使 arr1 中项的相对顺序和 arr2 中的相对顺序相同。未在 arr2 中出现过的元素需要按照升序放在 arr1 的末尾。

示例 1:

输入:arr1 = [2,3,1,3,2,4,6,7,9,2,19], arr2 = [2,1,4,3,9,6]
输出:[2,2,2,1,4,3,3,9,6,7,19]

示例  2:

输入:arr1 = [28,6,22,8,44,17], arr2 = [22,28,8,6]
输出:[22,28,8,6,17,44]

提示:

1 <= arr1.length, arr2.length <= 1000
0 <= arr1[i], arr2[i] <= 1000
arr2 中的元素 arr2[i]  各不相同 
arr2 中的每个元素 arr2[i] 都出现在 arr1 中

来源:力扣(LeetCode) 链接:leetcode-cn.com/problems/re… 著作权归领扣网络所有。

解题思路

根据题目意思,我们需要将 arr1 排序,排序的规则是:arr1 中的元素在 arr2 中存在则按 arr2 中的顺序排序,如果元素不在 arr2 中则按升序排序。

同时,因为 0 <= arr1[i], arr2[i] <= 1000,因此可以使用计数排序解决该问题:

  1. 对 arr1 进行计数,得到计数结果数组 cnt;
  2. 遍历 arr2 中的元素,向 arr1 中写入 cnt[x] 个 x,并且设置 cnt[x] 为 0;
  3. 遍历 cnt,如果 cnt[i] 不为 0,则向 arr1 中写入 cnt[i] 个 i,并且设置 cnt[i] 为 0;
  4. 返回 arr1;

代码实现

var relativeSortArray = function(arr1, arr2) {
    //定义计数数组并初始化元素为0
    const cnt = new Array(1001).fill(0)
    //对arr1进行统计
    for (const x of arr1) cnt[x] += 1
    
    let k = 0
    //将数组2中的元素放在前面
    for (const x of arr2) {
        //向数组1中写入 cnt[x] 个 x
        while(cnt[x]--) arr1[k++] = x
    }
    
    //将不在数组2中的元素放在末尾
    for (let i = 0; i < 1001; i++ ) {
        //数量小于1的元素跳过
        if (cnt[i] <= 0) continue
        //向数组1中写入 cnt[i] 个 i
        while(cnt[i]--) arr1[k++] = i
    }
    //返回已排序数组
    return arr1
};

如有错误欢迎指出,欢迎一起讨论!