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

[NOIP2003] 传染病控制 搜索+剪枝

时间:2017-08-09 20:02:10      阅读:162      评论:0      收藏:0      [点我收藏+]

标签:back   space   剪枝   noip2003   gis   return   stdin   cst   init   

搜索的最广泛应用优化——剪枝

这道题的dp和贪心都是无正确性的,所以,搜~~~~~~~

搜的时候你发现不剪枝极容易被卡掉(然而良心NOIP没有这么做,不剪枝仍然飞快),所以我们需要一些玄学的剪枝最常见的有俩:I.我们一层一层累加被感染人群若当前被感染人群已经大于已知解那么就舍去。 II.之后的每一层如果都取最大仍不如已知解优则舍去。

玄学.........

#include <cstdio>
#include <vector>
#include <algorithm>
#define MAXN 310
namespace pre{
  inline void read(int &sum){
    register char ch=getchar();
    for(sum=0;ch<0||ch>9;ch=getchar());
    for(;ch>=0&&ch<=9;sum=(sum<<1)+(sum<<3)+ch-0,ch=getchar());
  }
  inline int Max(int x,int y){
    return x>y?x:y;
  }
  int ans;
}
namespace tree{
  struct tree{
    int to,next;
  }c[MAXN<<1];
  int head[MAXN],t,size[MAXN],f[MAXN],Deep_Max,Max[MAXN];
  bool did[MAXN];
  std::vector<int> floor[MAXN];
  inline int comp(int x,int y){
    return size[x]>size[y];
  }
  inline void add(int x,int y){
    c[++t].to=y;
    c[t].next=head[x];
    head[x]=t;
  }
  inline void dfs(int x,int fa,int deep){
    Deep_Max=pre::Max(deep,Deep_Max);
    size[x]=1;
    f[x]=fa;
    floor[deep].push_back(x);
    for(int i=head[x];i;i=c[i].next)
      if(c[i].to!=fa)
        dfs(c[i].to,x,deep+1),size[x]+=size[c[i].to];
  }
  inline void Dfs(int deep,int ans){
    //if(ans+Max[deep+1]<=pre::ans)return;
    pre::ans=pre::Max(ans,pre::ans);
    if(deep==Deep_Max)return;
    for(int i=0;i<floor[deep+1].size();i++)
      if(did[f[floor[deep+1][i]]])
        did[floor[deep+1][i]]=1;
    for(int i=0;i<floor[deep+1].size();i++)
      if(did[floor[deep+1][i]]==0){
        did[floor[deep+1][i]]=1;
        Dfs(deep+1,ans+size[floor[deep+1][i]]);
        did[floor[deep+1][i]]=0;
      }
    for(int i=0;i<floor[deep+1].size();i++)
      if(did[f[floor[deep+1][i]]])
        did[floor[deep+1][i]]=0;
  }
}
namespace mid{
  int n,m;
  inline void Init(){
    using pre :: read;
    using tree :: add;
    read(n),read(m);
    for(int i=1,a,b;i<=m;i++)
      read(a),read(b),add(a,b),add(b,a);
    tree :: dfs(1,0,1);
  }
  inline void Work(){
    using namespace tree;
    for(int i=Deep_Max;i>1;i--){
      std::sort(floor[i].begin(),floor[i].end(),comp);
      Max[i]=Max[i+1]+size[floor[i][0]];
    }
    Dfs(1,0);
    printf("%d",n-pre::ans);
  }
}
int main(){
  freopen("epidemic.in","r",stdin);
  freopen("epidemic.out","w",stdout);
  mid :: Init();
  mid :: Work();
  return 0;
}

 

[NOIP2003] 传染病控制 搜索+剪枝

标签:back   space   剪枝   noip2003   gis   return   stdin   cst   init   

原文地址:http://www.cnblogs.com/TSHugh/p/7327175.html

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