返回介绍

solution / 0500-0599 / 0548.Split Array with Equal Sum / README

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

548. 将数组分割成和相等的子数组

English Version

题目描述

给定一个有 n 个整数的数组 nums ,如果能找到满足以下条件的三元组  (i, j, k)  则返回 true

  1. 0 < i, i + 1 < j, j + 1 < k < n - 1
  2. 子数组 (0, i - 1) , (i + 1, j - 1)(j + 1, k - 1)(k + 1, n - 1) 的和应该相等。

这里我们定义子数组 (l, r) 表示原数组从索引为 l 的元素开始至索引为 r 的元素。

 

示例 1: 

输入: nums = [1,2,1,2,1,2,1]
输出: True
解释:
i = 1, j = 3, k = 5. 
sum(0, i - 1) = sum(0, 0) = 1
sum(i + 1, j - 1) = sum(2, 2) = 1
sum(j + 1, k - 1) = sum(4, 4) = 1
sum(k + 1, n - 1) = sum(6, 6) = 1

示例 2:

输入: nums = [1,2,1,2,1,2,1,2]
输出: false

 

提示:

  • n == nums.length
  • 1 <= n <= 2000
  • -106 <= nums[i] <= 106

解法

方法一:前缀和 + 哈希表

先求出前缀和数组 s。

然后遍历 j 所有可能的位置,对于每个 j,找出 i,使得前两个子数组的和相等。同时将和添加到哈希表中。

接着对于每个 j,找出 k,使得后两个子数组的和相等,然后判断哈希表中是否存在该和,如果存在,则找到满足条件的三元组 (i, j, k),返回 true。

否则遍历结束返回 false。

时间复杂度 $O(n^2)$。

class Solution:
  def splitArray(self, nums: List[int]) -> bool:
    n = len(nums)
    s = [0] * (n + 1)
    for i, v in enumerate(nums):
      s[i + 1] = s[i] + v
    for j in range(3, n - 3):
      seen = set()
      for i in range(1, j - 1):
        if s[i] == s[j] - s[i + 1]:
          seen.add(s[i])
      for k in range(j + 2, n - 1):
        if s[n] - s[k + 1] == s[k] - s[j + 1] and s[n] - s[k + 1] in seen:
          return True
    return False
class Solution {
  public boolean splitArray(int[] nums) {
    int n = nums.length;
    int[] s = new int[n + 1];
    for (int i = 0; i < n; ++i) {
      s[i + 1] = s[i] + nums[i];
    }
    for (int j = 3; j < n - 3; ++j) {
      Set<Integer> seen = new HashSet<>();
      for (int i = 1; i < j - 1; ++i) {
        if (s[i] == s[j] - s[i + 1]) {
          seen.add(s[i]);
        }
      }
      for (int k = j + 2; k < n - 1; ++k) {
        if (s[n] - s[k + 1] == s[k] - s[j + 1] && seen.contains(s[n] - s[k + 1])) {
          return true;
        }
      }
    }
    return false;
  }
}
class Solution {
public:
  bool splitArray(vector<int>& nums) {
    int n = nums.size();
    vector<int> s(n + 1);
    for (int i = 0; i < n; ++i) s[i + 1] = s[i] + nums[i];
    for (int j = 3; j < n - 3; ++j) {
      unordered_set<int> seen;
      for (int i = 1; i < j - 1; ++i)
        if (s[i] == s[j] - s[i + 1])
          seen.insert(s[i]);
      for (int k = j + 2; k < n - 1; ++k)
        if (s[n] - s[k + 1] == s[k] - s[j + 1] && seen.count(s[n] - s[k + 1]))
          return true;
    }
    return false;
  }
};
func splitArray(nums []int) bool {
  n := len(nums)
  s := make([]int, n+1)
  for i, v := range nums {
    s[i+1] = s[i] + v
  }
  for j := 3; j < n-3; j++ {
    seen := map[int]bool{}
    for i := 1; i < j-1; i++ {
      if s[i] == s[j]-s[i+1] {
        seen[s[i]] = true
      }
    }
    for k := j + 2; k < n-1; k++ {
      if s[n]-s[k+1] == s[k]-s[j+1] && seen[s[n]-s[k+1]] {
        return true
      }
    }
  }
  return false
}

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

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

发布评论

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