• 第 113 场 LeetCode 双周赛题解


    A 使数组成为递增数组的最少右移次数

    在这里插入图片描述

    数据范围小直接模拟…

    class Solution {
    public:
        int minimumRightShifts(vector<int> &nums) {
            for (int op = 0; op < nums.size(); op++) {
                if (is_sorted(nums.begin(), nums.end()))//nums是否已经有序
                    return op;
                rotate(nums.begin(), prev(nums.end()), nums.end());//右移一次
            }
            return -1;
        }
    };
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    B 删除数对后的最小数组长度

    在这里插入图片描述
    在这里插入图片描述

    分类讨论:设数组中出现次数最多的数的出现次数为 m x mx mx

    1. 当数组长度为偶数时:若 m x ≤ n 2 mx \le \frac n 2 mx2n 则删除数对后的最小数组长度为 0 0 0 ,否则删除数对后的最小数组长度为 m x − ( n − m x ) mx - (n - mx) mx(nmx)
    2. 当数组长度为奇数时:若 m x ≤ ⌊ n 2 ⌋ mx \le \left \lfloor \frac n 2 \right \rfloor mx2n 则删除数对后的最小数组长度为 1 1 1 ,否则删除数对后的最小数组长度为 m x − ( n − m x ) mx - (n - mx) mx(nmx)
    class Solution {
    public:
        int minLengthAfterRemovals(vector<int> &nums) {
            unordered_map<int, int> cnt;//统计出现次数
            for (auto x: nums)
                cnt[x]++;
            int mx = 0;
            for (auto &[_, cnt_i]: cnt)
                mx = max(mx, cnt_i);
            int n = nums.size();
            if (n % 2 == 0) 
                return mx <= n / 2 ? 0 : mx - (n - mx);        
            return mx <= n / 2 ? 1 : mx - (n - mx);
        }
    };
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    C 统计距离为 k 的点对

    计数+枚举:枚举数组中的坐标 ( p [ 0 ] , p [ 1 ] ) (p[0],p[1]) (p[0],p[1]) ,枚举可能的与当前坐标距离为 k k k 的坐标:枚举 i ∈ [ 0 , k ] i \in [0,k] i[0,k] ( p [ 0 ] ∧ i , p [ 1 ] ∧ ( k − i ) ) (p[0]\wedge i, p[1]\wedge (k-i)) (p[0]i,p[1](ki)) 即为与当前坐标距离为 k k k 的坐标,若之前出现过这个坐标则更新答案,在枚举坐标 ( x , y ) (x,y) (x,y) 的过程中记录坐标的出现次数。

    class Solution {
    public:
        int countPairs(vector<vector<int>> &coordinates, int k) {
            map<pair<int, int>, int> cnt;//记录坐标的出现次数
            int res = 0;
            for (auto &p: coordinates) {
                for (int i = 0; i <= k; i++) {
                    int x = p[0] ^ i;
                    int y = p[1] ^ (k - i);
                    auto tmp = make_pair(x, y);
                    auto it = cnt.find(tmp);
                    if (it != cnt.end())//之前出现过坐标(x,y)
                        res += it->second;
                }
                cnt[{p[0], p[1]}]++;
            }
            return res;
        }
    };
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    D 可以到达每一个节点的最少边反转次数

    在这里插入图片描述
    在这里插入图片描述

    动态规划:首先建无向图,同时记录原始边的方向。不妨设 0 0 0 为树根,设 p [ c u r ] p[cur] p[cur] 为:使 c u r cur cur 能够到达以 c u r cur cur 为根的子树中的所有节点的最少边反转次数。通过 d f s dfs dfs 可以求出 p p p 数组。然后从 0 0 0 开始遍历树中节点 c u r cur cur,遍历过程中维护使 c u r cur cur 能够到达除以 c u r cur cur 为根的子树外的所有节点的最少边反转次数 u p _ c o s t up\_cost up_cost,则使 c u r cur cur 能够到达所有节点的最少边反转次数有 r e s [ c u r ] = p [ c u r ] + u p _ c o s t res[cur]=p[cur]+up\_cost res[cur]=p[cur]+up_cost

    class Solution {
    public:
        vector<int> minEdgeReversals(int n, vector<vector<int>> &edges) {
            vector<pair<int, int>> e[n];
            for (auto &ei: edges) {
                e[ei[0]].emplace_back(ei[1], 0);//0代表正向边
                e[ei[1]].emplace_back(ei[0], 1);//1代表反向边
            }
            int p[n];
            function<int(int, int)> dfs = [&](int cur, int par) {
                p[cur] = 0;
                for (auto &[j, w]: e[cur])
                    if (j != par) {
                        if (w == 1)//(cur,j)这条边需要反转
                            p[cur]++;
                        p[cur] += dfs(j, cur);
                    }
                return p[cur];
            };
            dfs(0, -1);//求p数组
            vector<int> res(n);
            function<void(int, int, int)> get = [&](int cur, int par, int up_cost) {
                res[cur] = p[cur] + up_cost;
                for (auto &[j, w]: e[cur])
                    if (j != par) {
                        if (w == 0)
                            get(j, cur, res[cur] - p[j] + 1);
                        else
                            get(j, cur, res[cur] - p[j] - 1);
                    }
            };
            get(0, -1, 0);//求答案数组
            return res;
        }
    };
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36

  • 相关阅读:
    从头开发一个RISC-V的操作系统(四)嵌入式开发介绍
    CSS选择器
    工控机连接Profinet转Modbus RTU网关与水泵变频器Modbus通讯配置案例
    Day06--下拉刷新
    机器人与AGI会撞出什么火花?
    【区块链 | 智能合约】Ethereum源代码(12)- 以太坊通过EVM执行交易过程分析
    python之字符串
    Windows系统下安装Ubuntu系统(双系统)
    面向对象分析与设计_用例图
    selenium京东商城爬取
  • 原文地址:https://blog.csdn.net/weixin_40519680/article/details/132927852