开源地址
为了便于大家学习,所有实现均已开源。欢迎 fork + star~
题目
给定一个二进制数组 nums , 找到含有相同数量的 0 和 1 的最长连续子数组,并返回该子数组的长度。
示例 1:
输入: nums = [0,1] 输出: 2 说明: [0, 1] 是具有相同数量 0 和 1 的最长连续子数组。 示例 2:
输入: nums = [0,1,0] 输出: 2 说明: [0, 1] (或 [1, 0]) 是具有相同数量0和1的最长连续子数组。
提示:
1 <= nums.length <= 10^5
nums[i] 不是 0 就是 1
v1-基本前缀和
思路
1) 构建好前缀好,方便计算子数组的和,是否为 k 的倍数。
2)因为只有 0 和1,所以和是数组长度的一半,说明二者一样多。
计算 i-j 的值,如果等于个数除以2,则满足。必须是偶数。且 i-j 的长度对应数组长度为偶数。
初步实现
class Solution {
public int findMaxLength(int[] nums) {
final int n = nums.length;
int[] prefix = new int[n];
prefix[0] = nums[0];
for(int i = 1; i < n; i++) {
prefix[i] = prefix[i-1] + nums[i];
}
// 从大大小遍历?
for(int step = n-1; step >=1; step--) {
int len = step+1;
if(len % 2 != 0) {
continue;
}
for(int i = 0; i < n - step; i++) {
int next = i + step;
int sum = prefix[next] - prefix[i] + nums[i];
if(sum == (len / 2)) {
return len;
}
}
}
return 0;
}
}
效果
1395ms 6.43%
勉强 AC
小结
这里用的是前缀和+暴力的解法。
比较好想到,但是暴力匹配确实性能比较差。
v2-改进
思路
我猜测有一种解法应该可以用位运算来实现。
可惜位运算一直比较菜。
TBC…
v3-前缀和+HashMap
思路
这一题不说和 T523 一模一样吧,只能说是一模一样。
数据处理
我们把 0 处理为 -1,1 保持不变。问题就变成了求 如何求得最长一段区间和为 0 的子数组
怎么求
sum[i] = sum[j]
那么
sum[i] - [sum]j = 0
根据前缀和定义,二者位置的数据就是等于 0 的。
代码
public int findMaxLength(int[] nums) {
final int n = nums.length;
int[] prefix = new int[n+1];
// 初始化
prefix[0] = nums[0] == 1 ? 1 : -1;
for(int i = 1; i < n; i++) {
// 预处理,简化计算
int val = nums[i] == 1 ? 1 : -1;
prefix[i] = prefix[i-1] + val;
}
Map<Integer, Integer> map = new HashMap<>();
// 处理从索引0开始的子数组
map.put(0, -1);
int maxLen = 0;
for(int i = 0; i < n; i++) {
int sum = prefix[i];
if(map.containsKey(sum)) {
maxLen = Math.max(maxLen, i - map.get(sum));
} else {
map.put(sum, i);
}
}
return maxLen;
}
效果
21ms 击败 77.14%
参考资料
https://leetcode.cn/problems/longest-well-performing-interval/submissions/578871050/?envType=problem-list-v2&envId=prefix-sum