码迷,mamicode.com
首页 > 其他好文 > 详细

P1120小木棍

时间:2019-09-11 20:13:39      阅读:76      评论:0      收藏:0      [点我收藏+]

标签:ref   space   就是   get   ++   大于等于   problem   很多   大于   

题目传送P1120

有许多剪枝的搜索

  • 先考虑答案的范围,先把小木棍排个序,原木棍的长一定大于等于最长的一根
  • 然后,就枚举一下原木棍的长度,搜是否能按这个长度拼成所有的木棍,如果能拼成直接就是答案了
  • 确定了原始长度,然后用总长度就能算出原始木棍的条数。如果不能整除的话,肯定不是答案
  • 然后确定搜索的状态,当前拼到第几跟,这一根还剩多长,拼这一根用的上一根小木棍的编号
  • 我们肯定要先拼大的,再用小的去补大的
  • 如果我们用小的凑的话,那些大的很难凑齐
  • 还有一个优化就是,有很多长度相同的小木棍,对于一种长度不能拼的话我们要直接略过剩下和它长度相同的。就预处理出$net$数组
  • 还有剪枝,如果这根小木棍正好等于剩下的长度,却没有成功。这是最优的情况都没有成功,说明这种情况不会合法了
  • 还有剪枝,如果这根小木棍等于原长却没被用上,说明这也是不合法的(这两个剪枝会剪掉4个点

    代码

#include<iostream>
#include<algorithm>
using namespace std;
inline int read(){
    int s=0,w=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();
    return s*w;
}
int n,x,num,tag,s,len;
int a[100],book[100],net[100];
bool cmp(int a,int b){
    return a>b;
}
void dfs(int now,int last,int res){
    if(!res){// now 当前是第几个,last拼的上一段的编号,res剩下的长度 
        if(now==num) {
            tag=1;
            return ;
        }
        
        for(int i=1;i<=n;i++)
          if(!book[i]){
            book[i]=1;
            dfs(now+1,i,len-a[i]);
            book[i]=0;
            if(tag) return ;
            break;
          }
    }
    int l=last+1,r=n,mid;
    while(l<=r){
        mid=(l+r)>>1;
        if(a[mid]>res) l=mid+1;
        else r=mid-1;
    }
    for(int i=l;i<=n;i++)
      if(!book[i]){
        book[i]=1;
        dfs(now,i,res-a[i]);
        book[i]=0;
        if(tag) return ;
        if(a[i]==res||res==len) return ;
        i=net[i];
        if(i==n) return ;
      }
}
int main()
{
    x=read();
    for(int i=1;i<=x;i++){
        int xx;
        xx=read();
        if(xx<=50) {
            a[++n]=xx;
            s+=xx;
        }
    }
    sort(a+1,a+n+1,cmp);
    for(int i=n;i>=1;i--)
      if(a[i]==a[i+1]) net[i]=net[i+1];
      else net[i]=i;
    for(len=a[1];len<=s/2;len++){
        if(s%len) continue;
        num=s/len;
        book[1]=1;
        tag=0;
        dfs(1,1,len-a[1]);
        book[1]=0;
        if(tag){printf("%d",len);return 0;}
    }
    printf("%d",s);
    return 0;
 } 

P1120小木棍

标签:ref   space   就是   get   ++   大于等于   problem   很多   大于   

原文地址:https://www.cnblogs.com/Vimin/p/11508254.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!