[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
};
相关推荐
simple_ssn20 分钟前
【C语言刷力扣】1502.判断能否形成等差数列
c语言·算法·leetcode
寂静山林28 分钟前
UVa 11855 Buzzwords
算法
Curry_Math33 分钟前
LeetCode 热题100之技巧关卡
算法·leetcode
ahadee40 分钟前
蓝桥杯每日真题 - 第10天
c语言·vscode·算法·蓝桥杯
军训猫猫头1 小时前
35.矩阵格式的一到一百数字 C语言
c语言·算法
Mr_Xuhhh2 小时前
递归搜索与回溯算法
c语言·开发语言·c++·算法·github
SoraLuna2 小时前
「Mac玩转仓颉内测版12」PTA刷题篇3 - L1-003 个位数统计
算法·macos·cangjie
爱吃生蚝的于勒4 小时前
C语言内存函数
c语言·开发语言·数据结构·c++·学习·算法
ChoSeitaku9 小时前
链表循环及差集相关算法题|判断循环双链表是否对称|两循环单链表合并成循环链表|使双向循环链表有序|单循环链表改双向循环链表|两链表的差集(C)
c语言·算法·链表
Fuxiao___10 小时前
不使用递归的决策树生成算法
算法