• poj 几道简单的搜索题(三)


    题目:poj 2531

    题意:

    给出n(n<=20)个点,及点与点之间的权值,把这n个点划分成两个集合,使这两个集合中点与点之间的边权和最小?

    分析:

    n=20,暴力枚举的时间复杂度才O(2^20*C),C是求和的时间常数。2000ms的时间足够了。暴力枚举的话可以子集枚举和递归(时间接近1000ms)。

    这题递归的话可以剪枝,可以优化到32ms,方法是参考:http://blog.csdn.net/martin31hao/article/details/8098302


    #include<cstdio>
    #include<cstring>
    using namespace std;
    typedef long long ll;
    const int N=22;
    int n,ans;
    int w[N][N];
    bool in[N];
    void dfs(int k,int sum)
    {
        in[k]=1;  //0表示在0集合,1表示在1集合
        int t=sum;
        for(int i=0;i<n;i++)
            if(in[i])t-=w[k][i];
            else t+=w[k][i];
        if(t>ans)ans=t;
        for(int i=k+1;i<n;i++)  //看下一个点
        if(t>sum){    //剪枝,只有t>sum才考虑以后的选择情况
            dfs(i,t); //把i放到1集合
            in[i]=0; //不放
        }
    }
    int main()
    {
        memset(in,0,sizeof(in));
        scanf("%d",&n);
        for(int i=0;i<n;i++)
            for(int j=0;j<n;j++)scanf("%d",&w[i][j]);
        ans=0;
        dfs(0,0);
        printf("%d
    ",ans);
        return 0;
    }
    题目:poj 1416

    题意:

    整数划分问题,要求把所给整数划分,使得划分得到的的数的和尽量接近一个目标数字。

    分析:

    枚举划分的个数,dfs分配划分的位置,然后把划分的数加起来求和即可。

    我是用一个string保存划分位置,最后再按照string保存的划分位置再求和。

    剪枝的话应该是每划分一次就把所得到的和,与未划分的数加起来,然后与已经得到最优解比较,判断是否剪枝即可。

    我的代码没优化,懒得改了QAQ。

    #include<cstdio>
    #include<iostream>
    #include<stack>
    #include<vector>
    #include<cstring>
    using namespace std;
    int n,ans,total,ansnum,len;
    vector<int>vec;
    string as;
    void dfs(int tot,int id,int k,string s)
    {
        if(tot==0){
            int i=0,sum=0;
            int pre=0;
            for(;i<=id;i++){
                int x=s[i]-'0';
                int j=x-1,w=1,num=0;
                while(j>=pre){
                    num+=vec[j--]*w;
                    w*=10;
                }
                pre=x;
                sum+=num;
            }
            if(sum>ans&&sum<=total)ans=sum,ansnum=1,as=s,len=id;
            else if(sum==ans)ansnum++;
            return;
        }
        for(int i=k+1;i<vec.size();i++){  //枚举划分的位置
            string t=s;
            t[id]=(char)(i+'0');
            dfs(tot-1,id+1,i,t);
        }
    }
    void change(int x)
    {
        stack<int>s;
        while(x){
            s.push(x%10);
            x/=10;
        }
        while(!s.empty())vec.push_back(s.top()),s.pop();
    }
    int main()
    {
        while(~scanf("%d%d",&total,&n)&&(total+n)){
            vec.clear();
            ans=-1;
            change(n);
            string s="1234567";
            for(int i=0;i<vec.size();i++){ //枚举划分的个数
                s[i]=(char)('0'+vec.size());
                dfs(i,0,0,s);
            }
            if(ans==-1)printf("error
    ");
            else if(ansnum>1)printf("rejected
    ");
            else{
                printf("%d",ans);
                int i=0,sum=0;
                int pre=0;
                for(;i<=len;i++){
                    int x=as[i]-'0';
                    int j=x-1,w=1,num=0;
                    while(j>=pre){
                        num+=vec[j--]*w;
                        w*=10;
                    }
                    pre=x;
                    printf(" %d",num);
                }
                printf("
    ");
            }
        }
        return 0;
    }
    

    题目:poj 2676

    题意:

    完成数独9*9,使每一行,每一列,每一小块3*3,都是由1-9组成

    分析:

    把没放数的位置先找出来,然后dfs一个个放数。用三个数组标记一下以放数的行,列和块。

    看Discuss说反搜更快。

    #include<cstdio>
    #include<cstring>
    #include<iostream>
    using namespace std;
    const int N=11;
    typedef pair<int,int>pii;
    pii q[N*N]; 
    bool cow[N][N],row[N][N],blo[N][N]; //列,行,块
    int a[N][N];
    int id[9][9]={0,0,0,1,1,1,2,2,2,0,0,0,1,1,1,2,2,2,0,0,0,1,1,1,2,2,2, //每个位置所属的块
                    3,3,3,4,4,4,5,5,5,3,3,3,4,4,4,5,5,5,3,3,3,4,4,4,5,5,5,
                    6,6,6,7,7,7,8,8,8,6,6,6,7,7,7,8,8,8,6,6,6,7,7,7,8,8,8};
    bool flag;
    int cnt;
    bool dfs(int s)
    {
        if(flag)return 1;
        if(s==cnt){
            flag=1;return 1;
        }
        bool ok=1;
        int r=q[s].first,c=q[s].second;
        for(int k=1;k<=9;k++){
            if(!row[r][k]&&!cow[c][k]&&!blo[id[r][c]][k]){
                a[r][c]=k; row[r][k]=cow[c][k]=blo[id[r][c]][k]=1;
                if(dfs(s+1))return 1;
               row[r][k]=cow[c][k]=blo[id[r][c]][k]=0;//a[r][c]=0;
            }
        }
        return 0;
    }
    int main()
    {
        int T;scanf("%d",&T);
        char s[11];
        while(T--){
            memset(row,0,sizeof(row));
            memset(cow,0,sizeof(cow));
            memset(blo,0,sizeof(blo));
            flag=0;
            cnt=0;
            for(int i=0;i<9;i++){
                scanf("%s",s);
                for(int j=0;j<9;j++){
                    a[i][j]=s[j]-'0';
                    int k=a[i][j];
                    if(k!=0)row[i][k] =cow[j][k] =blo[id[i][j]][k]=true;
                    else q[cnt++]=make_pair(i,j);
                }
            }
            dfs(0);
            for(int i=0;i<9;i++){
                for(int j=0;j<9;j++)printf("%d",a[i][j]);
                printf("
    ");
            }
    
        }
        return 0;
    }

    题目:poj 1129

    题意:

    图的染色问题,相邻点不能染同一种颜色,问需要最少颜色数?

    分析:

    dfs,依次枚举每个点,给他分配一种可以染的颜色,然后这种颜色从与它相邻的点的可染色中删除,然后再枚举下一个点,继续同样操作。

    #include<cstdio>
    #include<cstring>
    #include<vector>
    using namespace std;
    const int N=30;
    vector<int>g[N]; //存边
    int n;
    bool vis[N][N];
    void dfs(int u)
    {
        if(u==n)return;
        int i;
        for(i=0;i<n;i++){
            if(!vis[u][i]){vis[u][i]=1;break;} //找一种可染色
        }
        for(int j=0;j<g[u].size();j++){  
            vis[g[u][j]][i]=1;  //邻接点这种颜色不能用了
        }
        dfs(u+1);
    }
    char s[N];
    int main()
    {
        while(~scanf("%d",&n)&&n){
            for(int i=0;i<n;i++){
                scanf("%s",s);
                int len=strlen(s);
                g[s[0]-'A'].clear();
                if(len<=2)continue;
                for(int j=2;j<len;j++)
                    g[s[0]-'A'].push_back(s[j]-'A');
            }
            memset(vis,0,sizeof(vis));
            dfs(0);
            int ans=-1;
            for(int i=0;i<n;i++){
                int j;
                for(j=0;j<n;j++)if(!vis[i][j])break;
                if(j>ans)ans=j;
            }
            if(ans==1)printf("1 channel needed.
    ");
            else printf("%d channels needed.
    ",ans);
        }
        return 0;
    }
    









  • 相关阅读:
    Vue目录查询
    Vue框架学习(五)
    Vue框架学习(四)
    Vue框架学习(三)
    Vue框架学习(二)
    Vue框架学习(一)
    python中定义函数和参数的传递问题
    数据处理
    关于在程序中 文件新生成 在用os.system()程序对新生成的文件处理 举个栗子 如下:
    c++2008 并行配置文件和获取字典的所有key的方法
  • 原文地址:https://www.cnblogs.com/01world/p/5762840.html
Copyright © 2020-2023  润新知