[100天算法】-距离相等的条形码(day 59)

题目描述

复制代码
在一个仓库里,有一排条形码,其中第 i 个条形码为 barcodes[i]。

请你重新排列这些条形码,使其中两个相邻的条形码 不能 相等。 你可以返回任何满足该要求的答案,此题保证存在答案。

 

示例 1:

输入:[1,1,1,2,2,2]
输出:[2,1,2,1,2,1]
示例 2:

输入:[1,1,1,1,2,2,3,3]
输出:[1,3,1,3,2,1,2,1]
 

提示:

1 <= barcodes.length <= 10000
1 <= barcodes[i] <= 10000

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/distant-barcodes
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

方法1:直接排序

思路

  • 统计条形码的出现次数,按出现次数排序
  • 取出现次数最多的条形码,填充偶数位(0, 2, 4...)
  • 重复上一步骤直到偶数位填充完毕,然后开始填充奇数位(1, 3, 5...)

复杂度分析

  • 时间复杂度:O(NlogN),N 是 barcodes 的长度,统计条形码出现次数的时间是 O(N),排序时间是 O(klogk),k 是条形码总数,k 最坏情况下是 N。
  • 空间复杂度:O(N),哈希表的空间,最坏的情况是每个条形码都不一样。

代码

JavaScript Code

复制代码
/**
 * @param {number[]} barcodes
 * @return {number[]}
 */
var rearrangeBarcodes = function (barcodes) {
    const map = {};

    for (let i = 0; i < barcodes.length; i++) {
        const barcode = barcodes[i];
        map[barcode] = (map[barcode] || 0) + 1;
    }

    const list = Object.keys(map).map(b => [Number(b), map[b]]);
    list.sort((a, b) => a[1] - b[1])

    const res = Array(barcodes.length);
    let i = 0;

    while (list.length) {
        let [barcode, count] = list.pop();

        while (count-- > 0) {
            if (i >= barcodes.length) i = 1;

            res[i] = barcode;
            i += 2;
        }
    }
    return res;
};

方法2:堆排序

思路

  • 统计条形码的出现次数,建堆
  • 从堆中取出现次数最多的条形码,填充偶数位(0, 2, 4...)
  • 重复上一步骤直到偶数位填充完毕,然后开始填充奇数位(1, 3, 5...)

复杂度分析

  • 时间复杂度:O(NlogN),N 是 barcodes 的长度,统计条形码出现次数的时间是 O(N),每个条形码入堆出堆一次,时间是 O(NlogN)。
  • 空间复杂度:O(N),哈希表的空间。

代码

JavaScript Code

复制代码
/**
 * @param {number[]} barcodes
 * @return {number[]}
 */
var rearrangeBarcodes = function (barcodes) {
    const map = {};

    for (let i = 0; i < barcodes.length; i++) {
        const barcode = barcodes[i];
        map[barcode] = (map[barcode] || 0) + 1;
    }

    // 堆的数据结构 [barcode, count]
    const list = Object.keys(map).map(b => [Number(b), map[b]]);
    const heap = new MaxHeap(list, function comparator(inserted, compared) {
        return inserted[1] < compared[1];
    });

    const res = Array(barcodes.length);
    let i = 0;

    while (heap.size() > 0) {
        let [barcode, count] = heap.pop();

        while (count-- > 0) {
            if (i >= barcodes.length) i = 1;

            res[i] = barcode;
            i += 2;
        }
    }
    return res;
};

// **************************************************

class Heap {
    constructor(list = [], comparator) {
        this.list = list;
        this.comparator = comparator;

        this.init();
    }

    init() {
        const size = this.size();
        for (let i = Math.floor(size / 2) - 1; i >= 0; i--) {
            this.heapify(this.list, size, i);
        }
    }

    insert(n) {
        this.list.push(n);
        const size = this.size();
        for (let i = Math.floor(size / 2) - 1; i >= 0; i--) {
            this.heapify(this.list, size, i);
        }
    }

    peek() {
        return this.list[0];
    }

    pop() {
        const last = this.list.pop();
        if (this.size() === 0) return last;
        const returnItem = this.list[0];
        this.list[0] = last;
        this.heapify(this.list, this.size(), 0);
        return returnItem;
    }

    size() {
        return this.list.length;
    }
}

class MaxHeap extends Heap {
    constructor(list, comparator) {
        super(list, comparator);
    }

    heapify(arr, size, i) {
        let largest = i;
        const left = Math.floor(i * 2 + 1);
        const right = Math.floor(i * 2 + 2);

        if (left < size && this.comparator(arr[largest], arr[left]))
            largest = left;
        if (right < size && this.comparator(arr[largest], arr[right]))
            largest = right;

        if (largest !== i) {
            [arr[largest], arr[i]] = [arr[i], arr[largest]];
            this.heapify(arr, size, largest);
        }
    }
}

方法3

思路

  • 统计条形码的出现次数,建堆
  • 每次从堆中取两个出现次数最多的条形码,将它们加入排列结果中,然后数量分别减一后重新入堆
  • 直到堆中元素少于两个

复杂度分析

  • 时间复杂度:O(NlogN),N 是 barcodes 的长度,统计条形码出现次数的时间是 O(N),每个条形码入堆出堆一次,时间是 O(NlogN)。
  • 空间复杂度:O(N),哈希表的空间,最坏的情况是每个条形码都不一样。

代码

JavaScript Code

复制代码
/**
 * @param {number[]} barcodes
 * @return {number[]}
 */
var rearrangeBarcodes = function (barcodes) {
    const map = {};

    for (let i = 0; i < barcodes.length; i++) {
        const barcode = barcodes[i];
        map[barcode] = (map[barcode] || 0) + 1;
    }

    // 堆的数据结构 [barcode, count]
    const list = Object.keys(map).map(b => [Number(b), map[b]]);
    const heap = new MaxHeap(list, function comparator(inserted, compared) {
        return inserted[1] < compared[1];
    });

    const res = [];
    while (heap.size() > 1) {
        let [b1, cnt1] = heap.pop();
        let [b2, cnt2] = heap.pop();
        res.push(b1, b2)
        if (--cnt1 > 0) heap.insert([b1, cnt1])
        if (--cnt2 > 0) heap.insert([b2, cnt2])
    }
    if (heap.size()) {
        res.push(heap.pop()[0]);
    }
    return res
};
相关推荐
纽扣66729 分钟前
【算法进阶之路】链表进阶:删除、合并、回文与排序全解析
数据结构·算法·链表
消失的旧时光-19431 小时前
统一并发模型:线程、Reactor、协程本质是一件事(从线程到协程 · 第6篇·终章)
java·python·算法
智者知已应修善业1 小时前
【51单片机不用数组动态数码管显示字符和LED流水灯】2023-10-3
c++·经验分享·笔记·算法·51单片机
AI进化营-智能译站2 小时前
ROS2 C++开发系列16-智能指针管理传感器句柄|告别ROS2节点内存泄漏与野指针
java·c++·算法·ai
CS创新实验室2 小时前
从盘边到芯端——硬盘接口七十年变迁史
算法·磁盘调度
xvhao20133 小时前
单源、多源最短路
数据结构·c++·算法·深度优先·动态规划·图论·图搜索算法
MATLAB代码顾问3 小时前
多种群协同进化算法(MPCE)求解大规模作业车间调度问题——附MATLAB代码
开发语言·算法·matlab
FQNmxDG4S3 小时前
JVM内存模型详解:堆、栈、方法区与垃圾回收
java·jvm·算法
We་ct3 小时前
LeetCode 72. 编辑距离:动态规划经典题解
前端·算法·leetcode·typescript·动态规划
AI科技星3 小时前
精细结构常数α作为SI 7大基本量纲统一耦合常数的量子几何涌现理论
算法·机器学习·数学建模·数据挖掘·量子计算