返回介绍

solution / 1600-1699 / 1664.Ways to Make a Fair Array / README

发布于 2024-06-17 01:03:16 字数 5167 浏览 0 评论 0 收藏 0

1664. 生成平衡数组的方案数

English Version

题目描述

给你一个整数数组 nums 。你需要选择 恰好 一个下标(下标从 0 开始)并删除对应的元素。请注意剩下元素的下标可能会因为删除操作而发生改变。

比方说,如果 nums = [6,1,7,4,1] ,那么:

  • 选择删除下标 1 ,剩下的数组为 nums = [6,7,4,1] 。
  • 选择删除下标 2 ,剩下的数组为 nums = [6,1,4,1] 。
  • 选择删除下标 4 ,剩下的数组为 nums = [6,1,7,4] 。

如果一个数组满足奇数下标元素的和与偶数下标元素的和相等,该数组就是一个 平衡数组

请你返回删除操作后,剩下的数组_ _nums_ _是 平衡数组 的 方案数 。

 

示例 1:

输入:nums = [2,1,6,4]
输出:1
解释:
删除下标 0 :[1,6,4] -> 偶数元素下标为:1 + 4 = 5 。奇数元素下标为:6 。不平衡。
删除下标 1 :[2,6,4] -> 偶数元素下标为:2 + 4 = 6 。奇数元素下标为:6 。平衡。
删除下标 2 :[2,1,4] -> 偶数元素下标为:2 + 4 = 6 。奇数元素下标为:1 。不平衡。
删除下标 3 :[2,1,6] -> 偶数元素下标为:2 + 6 = 8 。奇数元素下标为:1 。不平衡。
只有一种让剩余数组成为平衡数组的方案。

示例 2:

输入:nums = [1,1,1]
输出:3
解释:你可以删除任意元素,剩余数组都是平衡数组。

示例 3:

输入:nums = [1,2,3]
输出:0
解释:不管删除哪个元素,剩下数组都不是平衡数组。

 

提示:

  • 1 <= nums.length <= 105
  • 1 <= nums[i] <= 104

解法

方法一:枚举 + 前缀和

我们先预处理得到数组 nums 的偶数下标元素之和 $s_1$ 以及奇数下标元素之和 $s_2$。

然后从前往后枚举数组 nums 的每个元素 $v$,用变量 $t_1$ 和 $t_2$ 分别记录已遍历的偶数下标元素之和以及奇数下标元素之和。

我们观察发现,对于当前遍历到的元素 $v$,如果删除了,那么该元素之后的奇偶下标元素之和会发生交换。此时,我们先判断该位置下标 $i$ 是奇数还是偶数。

  • 如果是偶数下标,删除该元素后,数组的奇数下标元素之和为 $t_2 + s_1 - t_1 - v$,而偶数下标元素之和为 $t_1 + s_2 - t_2$,如果这两个和相等,那么就是一个平衡数组,答案加一。

  • 如果是奇数下标,删除该元素后,数组的偶数下标元素之和为 $t_1 + s_2 - t_2 - v$,而奇数下标元素之和为 $t_2 + s_1 - t_1$,如果这两个和相等,那么就是一个平衡数组,答案加一。

然后我们更新 $t_1$ 和 $t_2$,继续遍历下一个元素。遍历完数组后,即可得到答案。

时间复杂度 $O(n)$,其中 $n$ 为数组的长度。空间复杂度 $O(1)$。

class Solution:
  def waysToMakeFair(self, nums: List[int]) -> int:
    s1, s2 = sum(nums[::2]), sum(nums[1::2])
    ans = t1 = t2 = 0
    for i, v in enumerate(nums):
      ans += i % 2 == 0 and t2 + s1 - t1 - v == t1 + s2 - t2
      ans += i % 2 == 1 and t2 + s1 - t1 == t1 + s2 - t2 - v
      t1 += v if i % 2 == 0 else 0
      t2 += v if i % 2 == 1 else 0
    return ans
class Solution {
  public int waysToMakeFair(int[] nums) {
    int s1 = 0, s2 = 0;
    int n = nums.length;
    for (int i = 0; i < n; ++i) {
      s1 += i % 2 == 0 ? nums[i] : 0;
      s2 += i % 2 == 1 ? nums[i] : 0;
    }
    int t1 = 0, t2 = 0;
    int ans = 0;
    for (int i = 0; i < n; ++i) {
      int v = nums[i];
      ans += i % 2 == 0 && t2 + s1 - t1 - v == t1 + s2 - t2 ? 1 : 0;
      ans += i % 2 == 1 && t2 + s1 - t1 == t1 + s2 - t2 - v ? 1 : 0;
      t1 += i % 2 == 0 ? v : 0;
      t2 += i % 2 == 1 ? v : 0;
    }
    return ans;
  }
}
class Solution {
public:
  int waysToMakeFair(vector<int>& nums) {
    int s1 = 0, s2 = 0;
    int n = nums.size();
    for (int i = 0; i < n; ++i) {
      s1 += i % 2 == 0 ? nums[i] : 0;
      s2 += i % 2 == 1 ? nums[i] : 0;
    }
    int t1 = 0, t2 = 0;
    int ans = 0;
    for (int i = 0; i < n; ++i) {
      int v = nums[i];
      ans += i % 2 == 0 && t2 + s1 - t1 - v == t1 + s2 - t2;
      ans += i % 2 == 1 && t2 + s1 - t1 == t1 + s2 - t2 - v;
      t1 += i % 2 == 0 ? v : 0;
      t2 += i % 2 == 1 ? v : 0;
    }
    return ans;
  }
};
func waysToMakeFair(nums []int) (ans int) {
  var s1, s2, t1, t2 int
  for i, v := range nums {
    if i%2 == 0 {
      s1 += v
    } else {
      s2 += v
    }
  }
  for i, v := range nums {
    if i%2 == 0 && t2+s1-t1-v == t1+s2-t2 {
      ans++
    }
    if i%2 == 1 && t2+s1-t1 == t1+s2-t2-v {
      ans++
    }
    if i%2 == 0 {
      t1 += v
    } else {
      t2 += v
    }
  }
  return
}
/**
 * @param {number[]} nums
 * @return {number}
 */
var waysToMakeFair = function (nums) {
  let [s1, s2, t1, t2] = [0, 0, 0, 0];
  const n = nums.length;
  for (let i = 0; i < n; ++i) {
    if (i % 2 == 0) {
      s1 += nums[i];
    } else {
      s2 += nums[i];
    }
  }
  let ans = 0;
  for (let i = 0; i < n; ++i) {
    const v = nums[i];
    ans += i % 2 == 0 && t2 + s1 - t1 - v == t1 + s2 - t2;
    ans += i % 2 == 1 && t2 + s1 - t1 == t1 + s2 - t2 - v;
    t1 += i % 2 == 0 ? v : 0;
    t2 += i % 2 == 1 ? v : 0;
  }
  return ans;
};

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文