Sticks(迭代搜索)

题意:(经典的迭代dfs搜索)有许多未知数目的相同木棍被“随意”的切割分成很多段,让你寻找出棍子可能存在的最小长度

思路:

通过题意理解,可能出现多种匹配的方法,最先想到的是进行dfs,但是对于此题最基本的dfs超时,这就需要我们使用迭代搜索,类似有bfs但是实则是一种更省时间的dfs来操作

通过访问数组中储存的被切过得棍子的长度来不断重建棍子,直到找到最终的棍子最佳长度和数目

代码:

#include<bits/stdc++.h>
using namespace std;
const int N=100;
int len[N];
bool vis[N];
int n,flag,sum,num;


bool cmp(int a,int b){
	return a>b;
}

bool dfs(int s,int cur,int cnt,int k){
	if(cnt==num){//当最后所得的棍子数刚好时,证明找到了最终的答案 
		return true;
	}
	for(int i=cur;i<n;i++){//依次枚举查找 
		if(vis[i]||(i&&len[i]==len[i-1]&&!vis[i-1])){ 
			continue;
		}
		if(len[i]+s==k){//当正好可以得到棍子的长度时 
			vis[i]=1;//该长度已经被使用,占上标记 
			//由于这一根棍子已经凑齐,下一次赢重新在一个棍子上进行操作,并重新开始搜索 
			if(dfs(0,0,cnt+1,k)){//如果在下一次循环完成了搜索 
				return true;//直接退出 
			}
			vis[i]=0;//回溯操作 
			return false;//若一直无法满足操作,证明不可能出现该长度的棍子,直接退出 
		}
		if(len[i]+s<k){//可以加上该长度,但是得到的总和仍不到最后棍子的长度,继续搜索 
			vis[i]=1;
			//继续对这根棍子进行搜索 
			if(dfs(len[i]+s,i+1,cnt,k)){
				return true;//直接退出 
			}
			vis[i]=0;
			//如果搜索之后回溯回来一直没有得到答案,直接返回错误 
			if(!s){
				return false;//如果 
			}
		}
	}
	return false;//若在大量的搜索下仍然没有找到,证明不存在,返回错误 
}

int main(){
	while(cin>>n&&n){
		sum=0;
		for(int i=0;i<n;i++){
//			cin>>len[i];	
			scanf("%d",&len[i]); 
			sum+=len[i];
		}
		sort(len,len+n,cmp);//进行降序排序 
		
		flag=0;
		for(int i=len[0];i<=sum/2;i++){//进行迭代搜索 
			if(sum%i==0){//因为需要找的棍子需是整数,必须可以整除与总数才行 
				memset(vis,false,sizeof(vis));
				num=sum/i;//棍子数量 
				if(dfs(0,0,0,i)){
					cout<<i<<endl;//得到最终值并且退出循环查找 
					flag=1;
					break;
				} 
			}
		}
		if(!flag){
			cout<<sum<<"\n";
		}
	}
	
	return 0;
} 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值