三个月前说学华为的理想汽车,裁员 18%

理想裁员

五一假期之前,就有小范围的爆料指出,理想汽车将大面积裁员。

这周该消息则正式落地:理想汽车宣布,组织大调整,整体优化比例超 18%

根据理想汽车 2023 年的财报显示,理想汽车有 3.16 万名员工,因此折算出来的数字近 5700 人。

可能有新来的网友会觉得:经济不景气,一个车企裁员有什么奇怪的?

首先,任何一家企业裁员超过 10% 都已经属于碰到大动脉了,而理想这一下子就是 18% 以上。

而更让我吃惊的是,这不是其他新能源车企,而是理想汽车。

我如果把前几个月,这公司的动向新闻重拿出来,大家就能感受到国内新能源的「内卷程度」和「不确定性」已经快进到了什么地步。

今年年初(2月底),理想因为「发放超额年终奖」上过热搜。

当时 8 个月年终奖到处是,15 个月年终奖的爆料也不少。

alt

理由很简单直接,因为业绩爆了:理想汽车 2023 全年实现营收 1238.5 亿元 同比增长 173.5%,年营收破千亿,实现自交付以来的首次健康盈利。

以至于理想 CEO 也发微博回应,说要学「华为」的利益分配,当时我也写过 文章 分析这事儿。

alt

而在 3 个月后的现在,这家公司宣布裁员 18%。

你可能会好奇,为什么短短三个月,一家公司变化会如此巨大?

在一个过度竞争的赛道,没有什么优势是永恒不变的,可能一个产品策略的失误,就能让一家公司直接倒闭,目前猜测让理想境况转折的点可能是:MEGA。

alt

大家觉得呢,是什么主导了理想这次的大规模裁员。

...

回归主题。

来一道和「华为」相关的算法原题。

题目描述

平台:LeetCode

题号:775

给你一个长度为 n 的整数数组 nums,表示由范围 内所有整数组成的一个排列。

全局倒置 的数目等于满足下述条件不同下标对 的数目:

局部倒置 的数目等于满足下述条件的下标 i 的数目:

当数组 nums 中 全局倒置 的数量等于 局部倒置 的数量时,返回 true ;否则,返回 false

示例 1:

输入:nums = [1,0,2]

输出:true

解释:有 1 个全局倒置,和 1 个局部倒置。

示例 2:

输入:nums = [1,2,0]

输出:false

解释:有 2 个全局倒置,和 1 个局部倒置。

提示:

  • nums 中的所有整数 互不相同
  • nums 是范围 内所有数字组成的一个排列

树状数组

根据题意,对于每个 而言:

  • 其左边比它大的 的个数,是以 为右端点的“全局倒置”数量,统计所有以 为右端点的“全局倒置”数量即是总的“全局倒置”数量 a

  • 同时我们可以将每个 与前一个值进行比较,从而统计总的“局部倒置”数量 b,其中 的取值范围为

一个容易想到的做法是利用「树状数组」,虽然该做法没有利用到核心条件「 是一个 的排列」,但根据数据范围 可知该复杂度为 的做法可过,且依赖的条件更少,适用范围更广。

Java 代码:

class Solution {
    int n;
    int[] tr;
    int lowbit(int x) {
        return x & -x;
    }
    void add(int x) {
        for (int i = x; i <= n; i += lowbit(i)) tr[i]++;
    }
    int query(int x) {
        int ans = 0;
        for (int i = x; i > 0; i -= lowbit(i)) ans += tr[i];
        return ans;
    }
    public boolean isIdealPermutation(int[] nums) {
        n = nums.length;
        tr = new int[n + 10];
        add(nums[0] + 1);
        int a = 0, b = 0;
        for (int i = 1; i < n; i++) {
            a += query(n) - query(nums[i] + 1);
            b += nums[i] < nums[i - 1] ? 1 : 0;
            add(nums[i] + 1);
        }
        return a == b;
    }
}

C++ 代码:

class Solution {
public:
    int n;
    vector<int> tr;
    int lowbit(int x) {
        return x & -x;
    }
    void add(int x) {
        for (int i = x; i <= n; i += lowbit(i)) tr[i]++;
    }
    int query(int x) {
        int ans = 0;
        for (int i = x; i > 0; i -= lowbit(i)) ans += tr[i];
        return ans;
    }
    bool isIdealPermutation(vector<int>& nums) {
        n = nums.size();
        tr.resize(n + 10);
        add(nums[0] + 1);
        long a = 0, b = 0;
        for (int i = 1; i < n; i++) {
            a += query(n) - query(nums[i] + 1);
            b += nums[i] < nums[i - 1] ? 1 : 0;
            add(nums[i] + 1);
        }
        return a == b;
    }
};

Python 代码:

class Solution:
    def lowbit(self, x):
        return x & -x

    def add(self, tr, x, n):
        i = x
        while i <= n:
            tr[i] += 1
            i += self.lowbit(i)

    def query(self, tr, x):
        i, ans = x, 0
        while i > 0:
            ans += tr[i]
            i -= self.lowbit(i)
        return ans

    def isIdealPermutation(self, nums: List[int]) -> bool:
        n = len(nums)
        tr = [0] * (n + 10)
        self.add(tr, nums[0] + 1, n)
        a, b = 00
        for i in range(1, n):
            a += self.query(tr, n) - self.query(tr, nums[i] + 1)
            b += nums[i] < nums[i - 1]
            self.add(tr, nums[i] + 1, n)
        return a == b
  • 时间复杂度:
  • 空间复杂度:

数学

解法一中并没有利用到核心条件「 是一个 的排列」,我们可以从该条件以及两类倒置的定义出发进行分析。

提示一:由“局部倒置”组成的集合为由“全局倒置”组成的集合的子集

任意一个“局部倒置”均满足“全局倒置”的定义,因此要判定两者数量是否相同,可转换为统计是否存在「不满足“局部倒置”定义的“全局倒置”」。

提示二:何为不满足“局部倒置”定义的“全局倒置”

结合题意,若存在坐标 ,满足 ,那么该倒置满足“全局倒置”定义,且不满足“局部倒置”定义。

若存在这样的逆序对,不满足,则有两类倒置数量不同。

提示三:考虑「如何构造」或「如何避免构造」不满足“全局倒置”定义的“局部倒置”

如果我们能够总结出「如何构造」或「如何避免构造」一个不满足“全局倒置”定义的“局部倒置” 所需的条件,问题可以转换为检查 nums 是否满足这样的条件,来得知 nums 是否存在不满足“全局倒置”定义的“局部倒置”。

我们可以结合「 是一个 的排列」来分析,若需要避免所有 的逆序对均不满足 ,只能是所有逆序对均由相邻数值产生。

Java 代码:

class Solution {
    public boolean isIdealPermutation(int[] nums) {
        for (int i = 0; i < nums.length; i++) {
            if (Math.abs(nums[i] - i) >= 2return false;
        }
        return true;
    }
}

C++ 代码:

class Solution {
public:
    bool isIdealPermutation(vector<int>& nums) {
        for (int i = 0; i < nums.size(); ++i) {
            if (abs(nums[i] - i) >= 2return false;
        }
        return true;
    }
};

Python 代码:

class Solution:
    def isIdealPermutation(self, nums: List[int]) -> bool:
        for i in range(len(nums)):
            if abs(nums[i] - i) >= 2:
                return False
        return True

TypeScript 代码:

function isIdealPermutation(nums: number[]): boolean {
    for (let i = 0; i < nums.length; i++) {
        if (Math.abs(nums[i] - i) >= 2return false;
    }
    return true;
};
  • 时间复杂度:
  • 空间复杂度:

最后

给大伙通知一下 📢 :

全网最低价 LeetCode 会员目前仍可用 ~

📅 年度会员:有效期加赠两个月!!; 季度会员:有效期加赠两周!!

🧧 年度会员:获 66.66 现金红包!!; 季度会员:获 22.22 现金红包!!

🎁 年度会员:参与当月丰厚专属实物抽奖(中奖率 > 30%)!!

专属链接:leetcode.cn/premium/?promoChannel=acoier

我是宫水三叶,每天都会分享算法知识,并和大家聊聊近期的所见所闻。

欢迎关注,明天见。

更多更全更热门的「笔试/面试」相关资料可访问排版精美的 合集新基地 🎉🎉

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值