R.I.P

最近几则关于打工人的新闻,十分让人痛心。

6 月 17 日早晨,科大讯飞智慧城市 BG 的一位同事在家突发不适,送往医院后经抢救无效去世,科大讯飞在 6 月 18 日内部通报此事。

同一天下午,一名 Shopee(跨境电商)深圳研发中心的同事突发不适,也是经抢救后无效,Shopee 于当天晚上 22 点内部通报此事。

最近,又开始连续有大厂员工猝死消息了_前端

R.I.P 🙏🏻🙏🏻🙏🏻

更具讽刺意味的是,前几天宁德时代还以实行 895 工作制,大干 100 天,外籍人员不强制冲上热搜。

虽然这事后来被辟谣:*宁德时代只是发出"奋斗 100 天"的号召,并没有强制员工 "895" 这样的规定。*提出这概念更多也只是为了目标明确,而非有生产压力,也与制裁因素无关。

但网友们对该解释并不买账,纷纷指责宁德时代鼓励员工内卷。

公司政策等外部因素,我们很难控制,能做的只能是明确自己承受边界在哪里。

身体机能是有预警机制的,遇到任何的身体不适,果断放下手上工作,及时休息,甚至就医。

不要被"责任心"和"职业素养"过分绑架,人在职场有时候是很身不由己,但也请你时刻记住上班的目的,是为了自己,为了家人,因此没有什么比健康更加重要。

希望我的读者大人们的都健健康康。

...

回归主线。

来一道和「设计数据结构」高度相关的题目。

题目描述

平台:LeetCode

题号:622

设计你的循环队列实现。

循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环,它也被称为“环形缓冲器”。

循环队列的一个好处是我们可以利用这个队列之前用过的空间。

在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。

但是使用循环队列,我们能使用这些空间去存储新的值。

你的实现应该支持如下操作:

  • MyCircularQueue(k): 构造器,设置队列长度为 最近,又开始连续有大厂员工猝死消息了_循环队列_02
  • Front: 从队首获取元素。如果队列为空,返回 最近,又开始连续有大厂员工猝死消息了_前端_03
  • Rear: 获取队尾元素。如果队列为空,返回 最近,又开始连续有大厂员工猝死消息了_前端_03
  • enQueue(value): 向循环队列插入一个元素。如果成功插入则返回真。
  • deQueue(): 从循环队列中删除一个元素。如果成功删除则返回真。
  • isEmpty(): 检查循环队列是否为空。
  • isFull(): 检查循环队列是否已满。

示例:

MyCircularQueue circularQueue = new MyCircularQueue(3); // 设置长度为 3
circularQueue.enQueue(1);  // 返回 true
circularQueue.enQueue(2);  // 返回 true
circularQueue.enQueue(3);  // 返回 true
circularQueue.enQueue(4);  // 返回 false,队列已满
circularQueue.Rear();  // 返回 3
circularQueue.isFull();  // 返回 true
circularQueue.deQueue();  // 返回 true
circularQueue.enQueue(4);  // 返回 true
circularQueue.Rear();  // 返回 4

提示:

  • 所有的值都在 最近,又开始连续有大厂员工猝死消息了_ci_05 至 最近,又开始连续有大厂员工猝死消息了_ci_06
  • 操作数将在 最近,又开始连续有大厂员工猝死消息了_面试_07最近,又开始连续有大厂员工猝死消息了_ci_06
  • 请不要使用内置的队列库。

数据结构

创建一个长度为 最近,又开始连续有大厂员工猝死消息了_循环队列_02 的数组充当循环队列,使用两个变量 heta 来充当队列头和队列尾(起始均为 最近,又开始连续有大厂员工猝死消息了_ci_05),整个过程 he 始终指向队列头部,ta 始终指向队列尾部的下一位置(待插入元素位置)。

两变量始终自增,通过与 最近,又开始连续有大厂员工猝死消息了_循环队列_02

分析各类操作的基本逻辑:

  • isEmpty 操作:当 heta 相等,队列存入元素和取出元素的次数相同,此时队列为空;
  • isFull 操作:ta - he 即队列元素个数,当元素个数为 最近,又开始连续有大厂员工猝死消息了_循环队列_02
  • enQueue 操作:若队列已满,返回 最近,又开始连续有大厂员工猝死消息了_前端_03,否则在 nums[ta % k] 位置存入目标值,并将 ta 指针后移;
  • deQueue 操作:若队列为空,返回 最近,又开始连续有大厂员工猝死消息了_前端_03,否则将 he 指针后移,含义为弹出队列头部元素;
  • Front 操作:若队列为空,返回 最近,又开始连续有大厂员工猝死消息了_前端_03,否则返回 nums[he % k] 队头元素;
  • Rear 操作:若队列为空,返回 最近,又开始连续有大厂员工猝死消息了_前端_03,否则返回 nums[(ta - 1) % k] 队尾元素;

Java 代码:

class MyCircularQueue {
    int k, he, ta;
    int[] nums;
    public MyCircularQueue(int _k) {
        k = _k;
        nums = new int[k];
    }
    public boolean enQueue(int value) {
        if (isFull()) return false;
        nums[ta % k] = value;
        return ++ta >= 0;
    }
    public boolean deQueue() {
        if (isEmpty()) return false;
        return ++he >= 0;
    }
    public int Front() {
        return isEmpty() ? -1 : nums[he % k];
    }
    public int Rear() {
        return isEmpty() ? -1 : nums[(ta - 1) % k];
    }
    public boolean isEmpty() {
        return he == ta;
    }
    public boolean isFull() {
        return ta - he == k;
    }
}

C++ 代码:

class MyCircularQueue {
public:
    int k, he, ta;
    vector<int> nums;
    MyCircularQueue(int _k) {
        k = _k;
        he = ta = 0;
        nums.resize(k);
    }
    bool enQueue(int value) {
        if (isFull()) return false;
        nums[ta % k] = value;
        return ++ta >= 0;
    }
    bool deQueue() {
        if (isEmpty()) return false;
        return ++he >= 0;
    }
    int Front() {
        return isEmpty() ? -1 : nums[he % k];
    }
    int Rear() {
        return isEmpty() ? -1 : nums[(ta - 1) % k];
    }
    bool isEmpty() {
        return he == ta;
    }
    bool isFull() {
        return ta - he == k;
    }
};

Python 代码:

class MyCircularQueue:
    def __init__(self, _k):
        self.k = _k
        self.he = self.ta = 0
        self.nums = [0] * _k

    def enQueue(self, value):
        if self.isFull(): return False
        self.nums[self.ta % self.k] = value
        self.ta += 1
        return self.ta >= 0

    def deQueue(self):
        if self.isEmpty(): return False
        self.he += 1
        return self.he >= 0

    def Front(self):
        return -1 if self.isEmpty() else self.nums[self.he % self.k]

    def Rear(self):
        return -1 if self.isEmpty() else self.nums[(self.ta - 1) % self.k]

    def isEmpty(self):
        return self.he == self.ta

    def isFull(self):
        return self.ta - self.he == self.k

TypeScript 代码:

class MyCircularQueue {
    k: number = 0; he: number = 0; ta: number = 0;
    nums: number[];
    constructor(k: number) {
        this.k = k
        this.nums = new Array<number>(this.k)
    }
    enQueue(value: number): boolean {
        if (this.isFull()) return false
        this.nums[this.ta % this.k] = value
        return this.ta++ >= 0
    }
    deQueue(): boolean {
        if (this.isEmpty()) return false
        return this.he++ >= 0
    }
    Front(): number {
        return this.isEmpty() ? -1 : this.nums[this.he % this.k]
    }
    Rear(): number {
        return this.isEmpty() ? -1 : this.nums[(this.ta - 1) % this.k]
    }
    isEmpty(): boolean {
        return this.he == this.ta
    }
    isFull(): boolean {
        return this.ta - this.he == this.k
    }
}
  • 时间复杂度:构造函数复杂度为 最近,又开始连续有大厂员工猝死消息了_面试_17,其余操作复杂度为 最近,又开始连续有大厂员工猝死消息了_前端_18
  • 空间复杂度:最近,又开始连续有大厂员工猝死消息了_面试_17