今天总算是碰到一道复杂题了。。。(我BT了?)
弄死我了,主要是排除重复情况,我想不通,最后参考了前人的代码,修改后AC了。。。
思路:通过递归,得出可能存在的解法,并记录,查找原记录,若存在相同则取消记录
#include <iostream>
#include <vector>
#include <string>
#include <stdlib.h>
using namespace std;

int sum;
int sticknum;
int stick[20];

int mark[20];

int solution;

int sols[100][20]; //记录不重复的答案
int slen[100]; //记录第j组答案的长度
int solnum; //记录不重复的答案的数量



void compute(int start,int cursum)
{

for(int i=start;i<sticknum;i++)
{

if(mark[i]==0&&stick[i]<=cursum)
{
mark[i]=1;
int tmp=cursum-stick[i];

if(tmp==0)
{
int tt=0;

for(int j=0;j<sticknum;j++)
{

if(mark[j]==1)
{
sols[solnum][tt++]=stick[j];
}
}

if(solution==0)
{ //如果是第一组答案,就直接记录
slen[solnum]=tt;
solution++;
solnum++;
mark[i]=0;

}else
{
bool test=false;

for(int k=0;k<solnum&&!test;k++)
{ //判断答案是否重复,如果重复就不对solnum加1,否则加1,

if(slen[k]==tt)
{ //先判断这两组组答案的长度是否相等,不等就直接判断下一组。

for(int m=0;m<tt;m++)
{
if(sols[k][m]!=sols[solnum][m])

{
break;
}
if(m==tt-1)test=true; //表示答案完全匹配,这一组答案就可以舍掉了
}
}
}

if(!test)
{ //如果答案与前面的都不重复,就对solution+1,solnum+1,
solution++;
slen[solnum]=tt;
solnum++;
}
}

}else
{
compute(i+1,tmp);
}
mark[i]=0;
}
}
}



int main()


{
while(1)

{
cin>>sum>>sticknum;
if(sum==0)break;
memset(stick,0,sizeof(stick));
memset(mark,0,sizeof(mark));
memset(sols,0,sizeof(sols));
memset(slen,0,sizeof(slen));
solution=0;
solnum=0;

for(int i=0;i<sticknum;i++)
{
cin>>stick[i];
}
cout<<"Sums of "<<sum<<":"<<endl;
compute(0,sum);

if(solution==0)
{
cout<<"NONE"<<endl;

}else
{
for(int i=0;i<solnum;i++)

{
cout<<sols[i][0];
for(int j=1;j<slen[i];j++)
cout<<"+"<<sols[i][j];
cout<<endl;
}
}
}
}