class Solution {
public:
int findLatestStep(vector<int>& arr, int m) {
}
};
1562. 查找大小为 M 的最新分组
给你一个数组 arr
,该数组表示一个从 1
到 n
的数字排列。有一个长度为 n
的二进制字符串,该字符串上的所有位最初都设置为 0
。
在从 1
到 n
的每个步骤 i
中(假设二进制字符串和 arr
都是从 1
开始索引的情况下),二进制字符串上位于位置 arr[i]
的位将会设为 1
。
给你一个整数 m
,请你找出二进制字符串上存在长度为 m
的一组 1
的最后步骤。一组 1
是一个连续的、由 1
组成的子串,且左右两边不再有可以延伸的 1
。
返回存在长度 恰好 为 m
的 一组 1
的最后步骤。如果不存在这样的步骤,请返回 -1
。
示例 1:
输入:arr = [3,5,1,2,4], m = 1 输出:4 解释: 步骤 1:"00100",由 1 构成的组:["1"] 步骤 2:"00101",由 1 构成的组:["1", "1"] 步骤 3:"10101",由 1 构成的组:["1", "1", "1"] 步骤 4:"11101",由 1 构成的组:["111", "1"] 步骤 5:"11111",由 1 构成的组:["11111"] 存在长度为 1 的一组 1 的最后步骤是步骤 4 。
示例 2:
输入:arr = [3,1,5,4,2], m = 2 输出:-1 解释: 步骤 1:"00100",由 1 构成的组:["1"] 步骤 2:"10100",由 1 构成的组:["1", "1"] 步骤 3:"10101",由 1 构成的组:["1", "1", "1"] 步骤 4:"10111",由 1 构成的组:["1", "111"] 步骤 5:"11111",由 1 构成的组:["11111"] 不管是哪一步骤都无法形成长度为 2 的一组 1 。
示例 3:
输入:arr = [1], m = 1 输出:1
示例 4:
输入:arr = [2,1], m = 2 输出:2
提示:
n == arr.length
1 <= n <= 10^5
1 <= arr[i] <= n
arr
中的所有整数 互不相同1 <= m <= arr.length
原站题解
cpp 解法, 执行用时: 168 ms, 内存消耗: 80.9 MB, 提交时间: 2023-09-19 10:33:46
class Solution { private: vector<int> father; void init(int n) { this->father = vector<int>(n + 1); iota(father.begin(), father.end(), 0); } // 必须路径压缩 int find(int x) { return x == father[x] ? x : father[x] = find(father[x]); } public: int findLatestStep(vector<int>& arr, int m) { int n = arr.size(); init(n); vector<int> cnt(n + 1); int ans = -1; int hasm = 0; for (int i = 0; i < n; i++) { // 本题的实际数据为[1, n] // 则作为哨兵的虚拟下标为0 int idx = arr[i]; int to = find(idx - 1); // 一轮循环影响两个状态位 // idx不用改变,但是名存实亡 // to会递增 if (cnt[idx] == m) { hasm --; } if (cnt[to] == m) { hasm --; } // merge 合并 合并到相邻位置 // 本题是idx-1的位置 father[idx] = to; cnt[to] += cnt[idx] + 1; // 新构造出的位置,为m则计数+1 if (cnt[to] == m) { hasm ++; } // 求最后存在的状态 if (hasm) { // ans存储第几步 ans = i + 1; } } return ans; } };
java 解法, 执行用时: 39 ms, 内存消耗: 52.7 MB, 提交时间: 2023-09-19 10:33:10
class Solution { public int findLatestStep(int[] arr, int m) { int n = arr.length; int[][] endpoints = new int[n + 1][2]; for (int i = 0; i <= n; i++) { Arrays.fill(endpoints[i], -1); } int cnt = 0; int ret = -1; for (int i = 0; i < n; i++) { int left = arr[i], right = arr[i]; if (arr[i] > 1 && endpoints[arr[i] - 1][0] != -1) { left = endpoints[arr[i] - 1][0]; int leftLength = endpoints[arr[i] - 1][1] - endpoints[arr[i] - 1][0] + 1; if (leftLength == m) { cnt--; } } if (arr[i] < n && endpoints[arr[i] + 1][1] != -1) { right = endpoints[arr[i] + 1][1]; int rightLength = endpoints[arr[i] + 1][1] - endpoints[arr[i] + 1][0] + 1; if (rightLength == m) { cnt--; } } int length = right - left + 1; if (length == m) { cnt++; } if (cnt > 0) { ret = i + 1; } endpoints[left][0] = endpoints[right][0] = left; endpoints[left][1] = endpoints[right][1] = right; } return ret; } }
python3 解法, 执行用时: 440 ms, 内存消耗: 31.3 MB, 提交时间: 2023-09-19 10:32:29
# 模拟 class Solution: def findLatestStep(self, arr: List[int], m: int) -> int: n = len(arr) endpoints = [(-1, -1) for _ in range(n + 1)] cnt = 0 ret = -1 for i in range(n): left = right = arr[i] if arr[i] > 1 and endpoints[arr[i] - 1][0] != -1: left = endpoints[arr[i] - 1][0] leftLength = endpoints[arr[i] - 1][1] - endpoints[arr[i] - 1][0] + 1; if leftLength == m: cnt -= 1 if arr[i] < n and endpoints[arr[i] + 1][1] != -1: right = endpoints[arr[i] + 1][1] rightLength = endpoints[arr[i] + 1][1] - endpoints[arr[i] + 1][0] + 1; if rightLength == m: cnt -= 1 length = right - left + 1 if length == m: cnt += 1 if cnt > 0: ret = i + 1 endpoints[left] = endpoints[right] = (left, right) return ret