LeetCode Weekly Contest 194

1486. 数组异或操作

给你两个整数,n 和 start 。

数组 nums 定义为:nums[i] = start + 2*i(下标从 0 开始)且 n == nums.length 。

请返回 nums 中所有元素按位异或(XOR)后得到的结果。

示例 1:

输入:n = 5, start = 0
输出:8
解释:数组 nums 为 [0, 2, 4, 6, 8],其中 (0 ^ 2 ^ 4 ^ 6 ^ 8) = 8 。
“^” 为按位异或 XOR 运算符。
示例 2:

输入:n = 4, start = 3
输出:8
解释:数组 nums 为 [3, 5, 7, 9],其中 (3 ^ 5 ^ 7 ^ 9) = 8.
示例 3:

输入:n = 1, start = 7
输出:7
示例 4:

输入:n = 10, start = 5
输出:2

提示:

1 <= n <= 1000
0 <= start <= 1000
n == nums.length

代码

class Solution {
    public int xorOperation(int n, int start) {
        int ans = start, i = 0;
        for (i=1; i<n; ++i) {
            ans ^= (start + 2 * i);
        }
        return ans;
    }
}

1487. 保证文件名唯一

题目难度Medium
给你一个长度为 n 的字符串数组 names 。你将会在文件系统中创建 n 个文件夹:在第 i 分钟,新建名为 names[i] 的文件夹。

由于两个文件 不能 共享相同的文件名,因此如果新建文件夹使用的文件名已经被占用,系统会以 (k) 的形式为新文件夹的文件名添加后缀,其中 k 是能保证文件名唯一的 最小正整数 。

返回长度为 n 的字符串数组,其中 ans[i] 是创建第 i 个文件夹时系统分配给该文件夹的实际名称。

示例 1:

输入:names = [“pes”,“fifa”,“gta”,“pes(2019)”]
输出:[“pes”,“fifa”,“gta”,“pes(2019)”]
解释:文件系统将会这样创建文件名:
“pes” --> 之前未分配,仍为 “pes”
“fifa” --> 之前未分配,仍为 “fifa”
“gta” --> 之前未分配,仍为 “gta”
“pes(2019)” --> 之前未分配,仍为 “pes(2019)”
示例 2:

输入:names = [“gta”,“gta(1)”,“gta”,“avalon”]
输出:[“gta”,“gta(1)”,“gta(2)”,“avalon”]
解释:文件系统将会这样创建文件名:
“gta” --> 之前未分配,仍为 “gta”
“gta(1)” --> 之前未分配,仍为 “gta(1)”
“gta” --> 文件名被占用,系统为该名称添加后缀 (k),由于 “gta(1)” 也被占用,所以 k = 2 。实际创建的文件名为 “gta(2)” 。
“avalon” --> 之前未分配,仍为 “avalon”
示例 3:

输入:names = [“onepiece”,“onepiece(1)”,“onepiece(2)”,“onepiece(3)”,“onepiece”]
输出:[“onepiece”,“onepiece(1)”,“onepiece(2)”,“onepiece(3)”,“onepiece(4)”]
解释:当创建最后一个文件夹时,最小的正有效 k 为 4 ,文件名变为 “onepiece(4)”。
示例 4:

输入:names = [“wano”,“wano”,“wano”,“wano”]
输出:[“wano”,“wano(1)”,“wano(2)”,“wano(3)”]
解释:每次创建文件夹 “wano” 时,只需增加后缀中 k 的值即可。
示例 5:

输入:names = [“kaido”,“kaido(1)”,“kaido”,“kaido(1)”]
输出:[“kaido”,“kaido(1)”,“kaido(2)”,“kaido(1)(1)”]
解释:注意,如果含后缀文件名被占用,那么系统也会按规则在名称后添加新的后缀 (k) 。

提示:

1 <= names.length <= 5 * 10^4
1 <= names[i].length <= 20
names[i] 由小写英文字母、数字和/或圆括号组成。

思路

哈希表。注意两个序列可能会发生冲突,注意区分,例如测试用例["kaido","kaido(1)","kaido","kaido(1)"]

代码

class Solution {
    public String[] getFolderNames(String[] names) {
        int n = names.length, i = 0;
        HashMap<String, Integer> nameMap = new HashMap<>();
        String[] ans = new String[n];
        for (i=0; i<n; ++i) {
            String name = names[i];
            if (nameMap.containsKey(name)) {
                int id = nameMap.get(name) + 1;
                String nameid = name + "(" + String.valueOf(id) + ")";
                while (nameMap.containsKey(nameid)) {
                    ++id;
                    nameid = name + "(" + String.valueOf(id) + ")";
                }
                nameMap.put(name, id);
                nameMap.put(nameid, 0);
                ans[i] = nameid;
            } else {
                ans[i] = name;
                nameMap.put(name, 0);
            }
        }
        return ans;
    }
}

1488. 避免洪水泛滥

你的国家有无数个湖泊,所有湖泊一开始都是空的。当第 n 个湖泊下雨的时候,如果第 n 个湖泊是空的,那么它就会装满水,否则这个湖泊会发生洪水。你的目标是避免任意一个湖泊发生洪水。

给你一个整数数组 rains ,其中:

rains[i] > 0 表示第 i 天时,第 rains[i] 个湖泊会下雨。
rains[i] == 0 表示第 i 天没有湖泊会下雨,你可以选择 一个 湖泊并 抽干 这个湖泊的水。
请返回一个数组 ans ,满足:

ans.length == rains.length
如果 rains[i] > 0 ,那么ans[i] == -1 。
如果 rains[i] == 0 ,ans[i] 是你第 i 天选择抽干的湖泊。
如果有多种可行解,请返回它们中的 任意一个 。如果没办法阻止洪水,请返回一个 空的数组 。

请注意,如果你选择抽干一个装满水的湖泊,它会变成一个空的湖泊。但如果你选择抽干一个空的湖泊,那么将无事发生(详情请看示例 4)。

示例 1:

输入:rains = [1,2,3,4]
输出:[-1,-1,-1,-1]
解释:第一天后,装满水的湖泊包括 [1]
第二天后,装满水的湖泊包括 [1,2]
第三天后,装满水的湖泊包括 [1,2,3]
第四天后,装满水的湖泊包括 [1,2,3,4]
没有哪一天你可以抽干任何湖泊的水,也没有湖泊会发生洪水。
示例 2:

输入:rains = [1,2,0,0,2,1]
输出:[-1,-1,2,1,-1,-1]
解释:第一天后,装满水的湖泊包括 [1]
第二天后,装满水的湖泊包括 [1,2]
第三天后,我们抽干湖泊 2 。所以剩下装满水的湖泊包括 [1]
第四天后,我们抽干湖泊 1 。所以暂时没有装满水的湖泊了。
第五天后,装满水的湖泊包括 [2]。
第六天后,装满水的湖泊包括 [1,2]。
可以看出,这个方案下不会有洪水发生。同时, [-1,-1,1,2,-1,-1] 也是另一个可行的没有洪水的方案。
示例 3:

输入:rains = [1,2,0,1,2]
输出:[]
解释:第二天后,装满水的湖泊包括 [1,2]。我们可以在第三天抽干一个湖泊的水。
但第三天后,湖泊 1 和 2 都会再次下雨,所以不管我们第三天抽干哪个湖泊的水,另一个湖泊都会发生洪水。
示例 4:

输入:rains = [69,0,0,0,69]
输出:[-1,69,1,1,-1]
解释:任何形如 [-1,69,x,y,-1], [-1,x,69,y,-1] 或者 [-1,x,y,69,-1] 都是可行的解,其中 1 <= x,y <= 10^9
示例 5:

输入:rains = [10,20,20]
输出:[]
解释:由于湖泊 20 会连续下 2 天的雨,所以没有没有办法阻止洪水。

提示:

1 <= rains.length <= 10^5
0 <= rains[i] <= 10^9’

思路

贪心。从前往后遍历每一天,在不下雨的时候,选择目前有水且之后最早下雨的湖泊把其中的水抽干。难点在于如何动态维护当前有水之后最早下雨两个集合:当前有水使用HashSet之后最早下雨使用最小堆。此题是一个比较综合的数据结构题,贪心的思想需要凭借良好的数据结构设计才能在较低的时间复杂度内实现。该实现的时间复杂度为O(nlogn),其中n = rains.length

代码

class Solution {
    class Lake implements Comparable<Lake> {
        public int num, occur;
        
        public Lake(int num, int occur) {
            this.num = num;
            this.occur = occur;
        }
        
        @Override
        public int compareTo(Lake other) {
            return this.occur - other.occur;
        }
    }
    
    public int[] avoidFlood(int[] rains) {
        int n = rains.length, i = 0, j = 0;
        HashSet<Integer> fulls = new HashSet<>();
        PriorityQueue<Lake> pq = new PriorityQueue<>();
        HashMap<Integer, ArrayList<Integer>> occurMap = new HashMap<>();
        HashMap<Integer, Integer> indexMap = new HashMap<>();
        for (i=0; i<n; ++i) {
            if (!occurMap.containsKey(rains[i])) {
                occurMap.put(rains[i], new ArrayList<Integer>());
            }
            occurMap.get(rains[i]).add(i);
            indexMap.put(rains[i], 0);
        }
        boolean success = true;
        int[] ans = new int[n];
        for (i=0; i<n; ++i) {
            if (rains[i] > 0) {
                if (fulls.contains(rains[i])) {
                    success = false;
                    break;
                }
                ans[i] = -1;
                fulls.add(rains[i]);
                indexMap.put(rains[i], indexMap.get(rains[i]) + 1);
                if (indexMap.get(rains[i]) < occurMap.get(rains[i]).size()) {
                    pq.add(new Lake(rains[i], occurMap.get(rains[i]).get(indexMap.get(rains[i]))));
                }
            } else {
                if (!pq.isEmpty()) {
                    Lake head = pq.poll();
                    ans[i] = head.num;
                    fulls.remove(head.num);
                } else {
                    ans[i] = 1;
                }
            }
        }
        if (!success) {
            ans = new int[0];
        } 
        return ans;
    }
}

1489. 找到最小生成树里的关键边和伪关键边

给你一个 n 个点的带权无向连通图,节点编号为 0 到 n-1 ,同时还有一个数组 edges ,其中 edges[i] = [fromi, toi, weighti] 表示在 fromi 和 toi 节点之间有一条带权无向边。最小生成树 (MST) 是给定图中边的一个子集,它连接了所有节点且没有环,而且这些边的权值和最小。

请你找到给定图中最小生成树的所有关键边和伪关键边。如果最小生成树中删去某条边,会导致最小生成树的权值和增加,那么我们就说它是一条关键边。伪关键边则是可能会出现在某些最小生成树中但不会出现在所有最小生成树中的边。

请注意,你可以分别以任意顺序返回关键边的下标和伪关键边的下标。

示例 1:
在这里插入图片描述

输入:n = 5, edges = [[0,1,1],[1,2,1],[2,3,2],[0,3,2],[0,4,3],[3,4,3],[1,4,6]]
输出:[[0,1],[2,3,4,5]]
解释:上图描述了给定图。
下图是所有的最小生成树。
在这里插入图片描述
注意到第 0 条边和第 1 条边出现在了所有最小生成树中,所以它们是关键边,我们将这两个下标作为输出的第一个列表。
边 2,3,4 和 5 是所有 MST 的剩余边,所以它们是伪关键边。我们将它们作为输出的第二个列表。
示例 2 :
在这里插入图片描述

输入:n = 4, edges = [[0,1,1],[1,2,1],[2,3,1],[0,3,1]]
输出:[[],[0,1,2,3]]
解释:可以观察到 4 条边都有相同的权值,任选它们中的 3 条可以形成一棵 MST 。所以 4 条边都是伪关键边。

提示:

2 <= n <= 100
1 <= edges.length <= min(200, n * (n - 1) / 2)
edges[i].length == 3
0 <= fromi < toi < n
1 <= weighti <= 1000
所有 (fromi, toi) 数对都是互不相同的。

思路

LeetCode上的第一道最小生成树 (MST) 题. 本题需要多次求解最小生成树,所以直接对边关于weight做排序,没有使用最小堆优化。由于题目与边有关,因此在求解Prim和Kruskal算法中选择Kruskal算法。Kruskal算法使用并查集优化,并查集使用路径压缩优化。
回到题目本身,『关键边』等价于去掉该边,则图不连通或最小生成树的边权和增大,『伪关键边』等价于去掉该边,则最小生成树的边权和不变(说明该边不是关键边),且以该边为起点用Kruskal算法求解最小生成树的边权和不变(说明至少有一个最小生成树包含该边)

代码

class Solution {
    class Edge implements Comparable<Edge> {
        public int from, to, weight, index;

        public Edge(int from, int to, int weight, int index) {
            this.from = from;
            this.to = to;
            this.weight = weight;
            this.index = index;
        }

        @Override
        public int compareTo(Edge other) {
            return this.weight - other.weight;
        }
    }

    class DisjointSets {
        int[] dsets;
        int[] ranks;

        public DisjointSets(int n) {
            dsets = new int[n];
            ranks = new int[n];
            for (int i=0; i<n; ++i) {
                dsets[i] = i;
            }
        }

        public int find(int x) {
            if (dsets[x] != x) {
                dsets[x] = find(dsets[x]);
            }
            return dsets[x];
        }

        public void union(int x, int y) {
            x = find(x);
            y = find(y);
            if (x != y) {
                if (ranks[x] > ranks[y]) {
                    dsets[y] = x;
                } else {
                    dsets[x] = y;
                    if (ranks[x] == ranks[y]) {
                        ++ranks[y];
                    }
                }
            }
        }
    }

    /**
     * if k == -1, use all edges
     * if return -1, graph not connected
     */
    private int KruskalWithoutEdgeK(ArrayList<Edge> edgeArr, int n, int k) {
        int cnt = 0, i = 0, cost = 0, m = edgeArr.size();
        DisjointSets djs = new DisjointSets(n);
        while (cnt < n - 1) {
            if (i == m) {
                return -1;
            }
            Edge curEdge = edgeArr.get(i++);
            if (k >= 0 && k == curEdge.index) {
                continue;
            }
            int r1 = djs.find(curEdge.from);
            int r2 = djs.find(curEdge.to);
            if (r1 != r2) {
                cost += curEdge.weight;
                djs.union(r1, r2);
                ++cnt;
            }
        }
        return cost;
    }

    private int KruskalStartsWithEdgeK(ArrayList<Edge> edgeArr, int n, Edge startEdge) {
        DisjointSets djs = new DisjointSets(n);
        int cnt = 1, i = 0, cost = startEdge.weight, m = edgeArr.size(), r1 = djs.find(startEdge.from), r2 = djs.find(startEdge.to);
        djs.union(r1, r2);   
        while (cnt < n - 1) {
            if (i == m) {
                return -1;
            }
            Edge curEdge = edgeArr.get(i++);
            if (startEdge.index == curEdge.index) {
                continue;
            }
            r1 = djs.find(curEdge.from);
            r2 = djs.find(curEdge.to);
            if (r1 != r2) {
                cost += curEdge.weight;
                djs.union(r1, r2);
                ++cnt;
            }
        }
        return cost;
    }

    public List<List<Integer>> findCriticalAndPseudoCriticalEdges(int n, int[][] edges) {
        ArrayList<List<Integer>> ans = new ArrayList<>();
        ans.add(new ArrayList<Integer>());
        ans.add(new ArrayList<Integer>());
        ArrayList<Edge> edgeArr = new ArrayList<>();
        int i = 0;
        for (int[] e: edges) {
            edgeArr.add(new Edge(e[0], e[1], e[2], i++));
        }
        Collections.sort(edgeArr);
        int mstCost = KruskalWithoutEdgeK(edgeArr, n, -1);
        for (Edge edge: edgeArr) {
            int curCost = KruskalWithoutEdgeK(edgeArr, n, edge.index);
            if (curCost == -1 || curCost > mstCost) {
                ans.get(0).add(edge.index);
            } else if (curCost == mstCost && KruskalStartsWithEdgeK(edgeArr, n, edge) == mstCost) {
                ans.get(1).add(edge.index);
            } 
        }
        return ans;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值