问题基本描述:有一个背包,能盛放的物品总重量为S,设有N件物品,其重量分别为w1,w2,...wn,希望从N件物品中选择若干件物品,所选物品的重量之和恰能放入该背包,即所选物品的重量之和等于S。
递归算法
#include
<
stdio.h
>
#define
N 7
#define
S 15
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
int
w[N
+
1
]
=
...
{0,1,2,3,4,5,6,7}
;
data:image/s3,"s3://crabby-images/42674/42674817e8fa3a8802ce8a04a989f75e48711c39" alt="“背包问题”的算法 “背包问题”的算法"
int
knap(
int
s,
int
n)
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
...
{
if(s == 0)
return 1;
if(s < 0 || s >0 && n < 1)
return 0;
if(knap(s - w[n], n-1))
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
printf("%4d ", w[n]);
return 1;
}
return knap(s, n-1);
}
void
main()
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
...
{
if(knap(S, N))
printf("OK! ");
else
printf("NO! ");
getchar();
}
非递归
#include
<
stdio.h
>
#define
N 7
#define
S 15
typedef
struct
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
...
{
int s;
int n;
int job;
}
KNAPTP;
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
int
w[N
+
1
]
=
...
{0,1,4,3,4,5,2,7}
;
data:image/s3,"s3://crabby-images/42674/42674817e8fa3a8802ce8a04a989f75e48711c39" alt="“背包问题”的算法 “背包问题”的算法"
int
knap(
int
s,
int
n)
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
...
{
KNAPTP stack[100], x;
int top, k, rep;
x.s = s; x.n = n;
x.job = 0;
top = 1; stack[top] = x;
k = 0;
while(top > 0 && k == 0)
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
x = stack[top];
rep = 1;
while(!k && rep)
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
if(x.s == 0)
k = 1; //caught an answer
else if(x.s < 0 || x.n <= 0)
rep = 0;
else
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
x.s = x.s - w[x.n--];
x.job = 1;
stack[++top] = x;
}
}
if(!k) //watch rep = 0;
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
rep =1;
while(top >= 1&& rep)
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
x = stack[top--];
if(x.job == 1)
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
x.s += w[x.n + 1];
x.job = 2; //change it as discard one
stack[++top] = x;
rep = 0;
}
}
}
}
if(k) // found an answer
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
while(top >= 1)
data:image/s3,"s3://crabby-images/c1580/c1580621f8694c15a3b1522215eecc9ee348f039" alt="“背包问题”的算法 “背包问题”的算法"
...{
x = stack[top--];
if(x.job == 1)
printf("%d ", w[x.n + 1]);
}
}
return k;
}
void
main()
data:image/s3,"s3://crabby-images/0851c/0851c82996fd01feb788b60094ba4195fd496cc9" alt="“背包问题”的算法 “背包问题”的算法"
...
{
if(knap(S, N))
printf("OK! ");
else
printf("NO ");
getchar();
}
求解部分的循环执行条件是堆栈不空且未找到解,因为当堆栈空的时候表示程序已经穷尽了所有的可能,这是使用堆栈的程序的一般特征。(利用堆栈对不在结果集中的数据进行回溯,堆栈中o的位置存放S和N,注意top的初始值是1并不是栈底),考虑数据int w[N+1] = {0,1,1,1,1,1,1,1};
堆栈不空的表达式应该是top>=1或者top>0,此处需要仔细体会stack[0]的内容和含义,否则我们很难正确构造堆栈不空的表达式;而未求得解表达式应该是k==0或者!k。