查找子集求和递归产生不正确的输出

时间:2015-06-27 15:29:52

标签: c algorithm recursion subset-sum

跟随this method (pdf)实施递归解决方案,但我认为它是错误的。我已按照pdf文件中的描述完全实现了它。

以下代码的输出错误。它显示了目标位置。目标数组包含应该添加以获得总和的数字的位表示。

binary[i] = 1置于else if条件可以解决问题,但这没有用。

是否有可能澄清什么是正确的解决方案?

#define SIZE(a) sizeof(a)/sizeof(a[0])
int binary[100];
void foo(int target, int i, int sum, int *a, int size) {
    binary[i] = 1;
    if (target == sum+a[i]) {
        int j;
        for (j=0;j<size;j++)
            printf("%d\n", binary[j]);
        return;
    } else if ((i+1 < size) && (sum + a[i] < target)) {
        foo(target, i+1, sum + a[i], a, size);
    }
    if ((i+1 < size) && (sum +a[i+1] <= target)) {
        binary[i] = 0;
        foo(target, i+1, sum, a, size);
    }
}

int main()
{
    int i, target =10,a[] = {1, 2, 3, 5, 100};
    foo(target, 0, 0, a, SIZE(a));
}

当前输出:0 1 1 1 1

预期输出:0 1 1 1 0

1 个答案:

答案 0 :(得分:0)

#include <stdio.h>

#define SIZE(a) sizeof(a)/sizeof(a[0])
int binary[100];

void show(int* p, int size) {
    int j;
    for (j = 0; j < size; j++)
        printf("%d\n", p[j]);
}

void foo(int target, int i, int sum, int *a, int size) {
    if (sum == target) {
        // solution found
        show(binary, size);
    } else if (sum < target && i < size) {
        // try both branches
        // 1. current value used
        binary[i] = 1;
        foo(target, i + 1, sum + a[i], a, size);
        // 2. current value skipped
        binary[i] = 0;
        foo(target, i + 1, sum, a, size);
    } else {
        // no solution on this path
    }
}

int main() {
    int target = 10;
    int a[] = {1, 2, 3, 5, 100};
    foo(target, 0, 0, a, SIZE(a));
}

说明:

  • 我们只谈isumbinary,因为从算法的角度来看,其他一切都是不变的。
  • 我们的目标是从a中选择总和等于target的单个元素。当我们达到这种情况时,我们就完成了,我们唯一的工作就是show解决方案。
  • 只有当前sum仍低于target且我们尚未到达a的末尾时,算法才会继续。
  • 总有两种可能:使用当前元素或跳过它。两者都是递归调查的。
  • 该程序必须在binary的适当位置留下零,否则会在backtracking的各个级别提供误导性结果。因此,首先调查 use 分支。