3SUM

来自testwiki
跳转到导航 跳转到搜索

Template:Unsolved计算复杂度理论中, 3SUM问题是指如下的问题:给定一个包含n个实数的集合,判断其中是否包含3个和为0的元素。问题也可以推广到一个更一般化的版本,rSUM,是要求判断集合中是否存在r个数的和为0。3SUM问题可以很容易地在O(n2)的时间复杂度内解决。对于某些特化的计算模型,这已经达到了它们Ω(nr/2)的复杂度下界。

人们曾经猜想任何3SUM问题的确定性算法都至少需要Ω(n2)的时间复杂度。然而在2014年,最初的3SUM被Allan Grønlund和Seth Pettie否决了。他们给出了一个能在能在O(n2/(logn/loglogn)2/3)的时间复杂度内求解3SUM问题的确定性算法。[1]目前仍然有人猜想3SUM是不可能在O(n2Ω(1))的时间复杂度内解决的。

二次时间复杂度算法

设输入数据存储与数组S[0..n1],3SUM问题可以通过散列表在平均O(n2)的时间复杂度内解决。方法是先将S中的元素全部插入到一个散列表中,然后对于每一个数组下标对(i,j),检查(S[i]+S[j])是否在散列表中即可。

一个更好的方法是,先将输入数据排序,然后用一种巧妙的方法测试所有可能的输入组合,而避免使用二分查找。这种办法即使在最坏情况下也可以保证O(n2)的时间复杂度,它的伪代码如下所示:[2]

 sort(S);
 for i=0 to n-3 do
    a = S[i];
    start = i+1;
    end = n-1;
    while (start < end) do
       b = S[start]
       c = S[end];
       if (a+b+c == 0) then
          output a, b, c;
          // Continue search for all triplet combinations summing to zero.
           end = end - 1
       else if (a+b+c > 0) then
          end = end - 1;
       else
          start = start + 1;
       end
    end
 end

下面的例子展示了算法在一个较小的数组上是如何执行的。变量a的当前值以绿色标记,而bc的当前值以红色标记。

 -25 -10 -7 -3 2 4 8 10  (a+b+c==-25)
 -25 -10 -7 -3 2 4 8 10  (a+b+c==-22)
 . . .
 -25 -10 -7 -3 2 4 8 10  (a+b+c==-7)
 -25 -10 -7 -3 2 4 8 10  (a+b+c==-7)
 -25 -10 -7 -3 2 4 8 10  (a+b+c==-3)
 -25 -10 -7 -3 2 4 8 10  (a+b+c==2)
 -25 -10 -7 -3 2 4 8 10  (a+b+c==0)

我们可以从上面的过程中看出这个算法为什么是正确的。假设3SUM问题有一组解a + b + c = 0。首先单向移动的最左侧下标必然会到达a,而之后剩下的两个下标会有一个先遇到b或者c,而根据a+b+c > 0时移动右侧下标,反之移动左侧下标的规则,在此之后先遇到的下标会保持不动,直到我们移动另外一个下标找到对应的那一组解为止。因此对于3SUM问题的任意一个解最终都会被这个算法发现。

变体

总和非零

用下列方法可以搜索出集合中和为任意常数C的三个元素:

  • 将集合中每个元素分别减去C/3;
  • 对新集合使用3SUM算法求解。

三个不同数组

我们也能从三个整数集合中分别找出一个元素使其和为零,即对于给定的三个整数集合X、Y、Z,找出三个数Template:Math使得Template:Tmath。下文记单集合的3SUM问题为3SUM×1,三集合的3SUM问题为3SUM×3,则3SUM×3按下列步骤即可转化为3SUM×1:

根据第一步中的变换,一定有Template:Math[3]


参考资料

Template:Reflist