poj1011(搜索应用)

        唉,做的挺纠结的一题,参考了大牛们的代码,最后发现这个解法最易懂,就拿出来和大家分享下!!!

          题目大意及解题思想:

         【题意】:乔治拿来一组等长的木棒,将它们随机地砍断,使得每一节木棍的长度都不超过50个长度单位。然后他又想把这些木棍恢复到为裁截前的状态,但忘记了初始时有多少木棒以及木棒的初始长度。请你设计一个程序,帮助乔治计算木棒的可能最小长度。每一节木棍的长度都用大于零的整数表示。

         【题解】:不得不说,这道题出得非常好,特别是uva那里的大数据(poj的数据太水了),对于剪枝能力要求很高。下面说下几个重要的剪枝:
1.把所有木棍的长度从大到小排列,组合木棒时优先使用长的木棍,这样可以加快组合速度,并且对后面的剪枝有帮助。
2.木棒的长度一定是大于等于最长木棍的长度并且小于等于所有木棍长度的和,这个很容易证明。
3.木棒的长度一定是所有木棍长度的和的约数,这个也很容易证明。
4.在某一个木棒的组合过程中,对于当前的木棍stick[i],如果stick[i-1]没有被组合并且stick[i] == stick[i-1],那么不用考虑stick[i],显然stick[i]最终也不会被组合。
5.如果此次是在尝试第i个木棒的第一段,假设stick[j]为当前可以被使用的最长的木棍,如果此次组合失败,直接退出搜索,即退回到对第i-1个木棒的搜索。试想:失败说明现在使用stick[j]是不可行的,那么以后无论什么时候使用stick[j]都是不可行的,因为以后再处理stick[j]时可使用的木棍一定是当前可使用的木棍的子集,在更多木棍选择的情况下都不能组合成功,那么,在更少木棍选择的情况下一定不能组合成功。

不多说,代码如下:

      

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int N,minlen,sum,num;
int sticks[65],used[65];
int cmp(const void *a,const void *b)
{
	return *(int*)b-*(int*)a;
}
int dfs(int n,int len,int pos)
{
	int i;
	if(n==num)
		return 1;
	for(i=pos;i<N;i++)
	{
		if(used[i]||(i&&!used[i-1]&&sticks[i]==sticks[i-1]))continue;
	    if(len+sticks[i]==minlen)
		{
			used[i]=1;
			if(dfs(n+1,0,0))
				return 1;
			used[i]=0;
			return 0;
		}
    	else if(len+sticks[i]<minlen)
		{
			used[i]=1;
			if(dfs(n,len+sticks[i],i+1))
				return 1;
			used[i]=0;
            if(len==0)return 0;
		}
	}
	return 0;
}
int main()
{
	while(scanf("%d",&N)!=EOF&&N!=0)
	{
       int i,end=0,sum=0;
	   for(i=0;i<N;i++)
	   {
		   scanf("%d",&sticks[i]);
		   sum+=sticks[i];
	   }
	       qsort(sticks,N,sizeof(int),cmp);
           memset(used,0,sizeof(used));
        for(minlen=sticks[0];minlen<=sum;minlen++)
		{ 
			
			if(sum%minlen==0)
			{
				  num=sum/minlen;
				  if(dfs(0,0,0))
				  {
					  end=1;
					  printf("%d\n",minlen);
					  break;
				  }
			}
		}
		if(!end)
			printf("%d\n",sum);
	}
	return 0;
}


~_~

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值