zoukankan      html  css  js  c++  java
  • Cogs 2221. [SDOI2016 Round1] 数字配对

    2221. [SDOI2016 Round1] 数字配对

    题目链接

    ★★★   输入文件:menci_pair.in   输出文件:menci_pair.out   简单对比
    时间限制:1 s   内存限制:128 MB

    【题目描述】

    有 n 种数字,第 i 种数字是 ai、有 bi 个,权值是 ci

    若两个数字 aiaj 满足,ai 是 aj 的倍数,且 aiaj 是一个质数,那么这两个数字可以配对,并获得 ci×cj 的价值。

    一个数字只能参与一次配对,可以不参与配对。
    在获得的价值总和不小于 0 的前提下,求最多进行多少次配对。

    【输入格式】

    第一行一个整数 n
    第二行 n 个整数 a1a2、……、an
    第三行 n 个整数 b1b2、……、bn
    第四行 n 个整数 c1c2、……、cn

    【输出格式】

    一行一个数,最多进行多少次配对。

    【样例输入】

    3

    2 4 8

    2 200 7

    -1 -2 1

    【样例输出】

    4

    【提示】

    测试点 1 ~ 3:n10ai109bi=1ci∣≤105
    测试点 4 ~ 5:n200ai109bi105ci=0
    测试点 6 ~ 10:n200ai109bi105ci∣≤105

    【来源】

    SDOI2016 Round1 Day1

    #include<iostream>
    #include<cstdio>
    #include<queue>
    #define maxn 501
    #define INF 1e9
    using namespace std;
    int n,a[maxn],b[maxn],c[maxn],ans,head[maxn],cur[maxn],lev[maxn],t,num=1;
    struct node{
        int to,pre,cap;
    }e[maxn*10000];
    void Insert(int from,int to,int v){
        e[++num].to=to;
        e[num].cap=v;
        e[num].pre=head[from];
        head[from]=num;
    }
    bool flag1,flag2,vis[maxn];
    bool pan(int x,int y){
        if(x<y)swap(x,y);
        int z=x/y;
        if(x%y!=0)return 0;
        for(int i=2;i*i<=z;i++){
            if(z%i==0)return 0;
        }
        return 1;
    }
    void dfs(int pos,int cost,int cnt){
        if(cost>=0)ans=max(ans,cnt);
        if(pos==n+1)return;
        if(vis[pos]){
            dfs(pos+1,cost,cnt);
            return;
        }
        for(int i=1;i<=n;i++){
            if(!vis[i]&&i!=pos&&pan(a[i],a[pos])){
                vis[i]=1;vis[pos]=1;
                dfs(pos+1,cost+c[pos]*c[i],cnt+1);
                dfs(pos+1,cost,cnt);
                vis[i]=0;vis[pos]=0;
            }
        }
    }
    void jiantu(){
        for(int i=1;i<=n;i++){
            Insert(0,i,b[i]);
            Insert(i,0,0);
        }
        for(int i=1;i<=n;i++){
            Insert(i+n,t,b[i]);
            Insert(t,i+n,0);
        }
        for(int i=1;i<=n;i++)
            for(int j=1;j<=n;j++){
                if(a[i]<=a[j]||i==j)continue;
                if(pan(a[i],a[j])){
                    Insert(i,j+n,INF);
                    Insert(j+n,i,0);
                }
            }
    }
    bool bfs(){
        queue<int>q;q.push(0);
        for(int i=0;i<=n*2+1;i++)cur[i]=head[i],lev[i]=-1;
        lev[0]=0;
        while(!q.empty()){
            int now=q.front();q.pop();
            for(int i=head[now];i;i=e[i].pre){
                int to=e[i].to;
                if(lev[to]==-1&&e[i].cap>0){
                    lev[to]=lev[now]+1;
                    q.push(to);
                    if(to==t)return 1;
                }
            }
        }
        return 0;
    }
    int dinic(int now,int flow){
        if(now==t)return flow;
        int rest=0,delta;
        for(int &i=cur[now];i;i=e[i].pre){
            int to=e[i].to;
            if(e[i].cap>0&&lev[to]==lev[now]+1){
                delta=dinic(to,min(e[i].cap,flow-rest));
                if(delta){
                    e[i].cap-=delta;e[i^1].cap+=delta;
                    rest+=delta;
                    if(rest==flow)break;
                }
            }
        }
        if(rest!=flow)lev[now]=-1;
        return rest;
    }
    int main(){
        freopen("Cola.txt","r",stdin);
    //    freopen("menci_pair.in","r",stdin);freopen("menci_pair.out","w",stdout);
        scanf("%d",&n);
        for(int i=1;i<=n;i++)scanf("%d",&a[i]);
        for(int i=1;i<=n;i++){
            scanf("%d",&b[i]);
            if(b[i]!=1)flag1=1;
        }
        for(int i=1;i<=n;i++){
            scanf("%d",&c[i]);
            if(c[i]!=0)flag2=1;
        }
        if(!flag1&&n<=10){//1~3 
            dfs(1,0,0);
            printf("%d",ans);
            return 0;
        }
        if(!flag2){//4~5
            t=n*2+1;
            jiantu();
            while(bfs())ans+=dinic(0,INF);
            printf("%d",ans);
        }
    }
    写的50分代码 4,5个点挂了
    #include<iostream>
    #include<cstdio>
    #include<queue>
    #define maxn 501
    #define INF 1e9
    using namespace std;
    int n,a[maxn],b[maxn],c[maxn],ans,head[maxn],cur[maxn],lev[maxn],t,num=1;
    struct node{
        int to,pre,cap;
    }e[maxn*10000];
    void Insert(int from,int to,int v){
        e[++num].to=to;
        e[num].cap=v;
        e[num].pre=head[from];
        head[from]=num;
    }
    bool flag1,flag2,vis[maxn];
    bool pan(int x,int y){
        if(x<y)swap(x,y);
        int z=x/y;
        if(x==y)
            return 0;
        if(x%y!=0)return 0;
        for(int i=2;i*i<=z;i++){
            if(z%i==0)return 0;
        }
        return 1;
    }
    void dfs(int pos,int cost,int cnt){
        if(cost>=0)ans=max(ans,cnt);
        if(pos==n+1)return;
        if(vis[pos]){
            dfs(pos+1,cost,cnt);
            return;
        }
        for(int i=1;i<=n;i++){
            if(!vis[i]&&i!=pos&&pan(a[i],a[pos])){
                vis[i]=1;vis[pos]=1;
                dfs(pos+1,cost+c[pos]*c[i],cnt+1);
                dfs(pos+1,cost,cnt);
                vis[i]=0;vis[pos]=0;
            }
        }
    }
    void jiantu(){
        for(int i=1;i<=n;i++){
            Insert(0,i,b[i]);
            Insert(i,0,0);
        }
        for(int i=1;i<=n;i++){
            Insert(i+n,t,b[i]);
            Insert(t,i+n,0);
        }
        for(int i=1;i<=n;i++)
            for(int j=1;j<=n;j++){
                if(i==j)continue;
                if(pan(a[i],a[j])){
                    Insert(i,j+n,INF);
                    Insert(j+n,i,0);
                }
            }
    }
    bool bfs(){
        queue<int>q;q.push(0);
        for(int i=0;i<=n*2+1;i++)cur[i]=head[i],lev[i]=-1;
        lev[0]=0;
        while(!q.empty()){
            int now=q.front();q.pop();
            for(int i=head[now];i;i=e[i].pre){
                int to=e[i].to;
                if(lev[to]==-1&&e[i].cap>0){
                    lev[to]=lev[now]+1;
                    q.push(to);
                    if(to==t)return 1;
                }
            }
        }
        return 0;
    }
    int dinic(int now,int flow){
        if(now==t)return flow;
        int rest=0,delta;
        for(int &i=cur[now];i;i=e[i].pre){
            int to=e[i].to;
            if(e[i].cap>0&&lev[to]==lev[now]+1){
                delta=dinic(to,min(e[i].cap,flow-rest));
                if(delta){
                    e[i].cap-=delta;e[i^1].cap+=delta;
                    rest+=delta;
                    if(rest==flow)break;
                }
            }
        }
        if(rest!=flow)lev[now]=-1;
        return rest;
    }
    int main(){
        //freopen("Cola.txt","r",stdin);
        freopen("menci_pair.in","r",stdin);freopen("menci_pair.out","w",stdout);
        scanf("%d",&n);
        for(int i=1;i<=n;i++)scanf("%d",&a[i]);
        for(int i=1;i<=n;i++){
            scanf("%d",&b[i]);
            if(b[i]!=1)flag1=1;
        }
        for(int i=1;i<=n;i++){
            scanf("%d",&c[i]);
            if(c[i]!=0)flag2=1;
        }
        if(!flag1&&n<=10){//1~3 
            dfs(1,0,0);
            printf("%d",ans);
            return 0;
        }
        if(!flag2){//4~5
            t=n*2+1;
            jiantu();
            while(bfs())ans+=dinic(0,INF);
            printf("%d",ans/2);
        }
        return 0;
    }
    50分 爆搜30+最大流20
    /*
        根据题意,如果我们把a[i]质因数分解,那么如果x,y能够建边,那么它们分解出来的个数一定相差1,这样就转成了二分图。
        至于题目要求的保证费用要大于等于0,也就是越大越好,我们可以将费用变负,然后跑最小费用,每次增广保证费用不大于0。
    */
    #include<iostream>
    #include<cstdio>
    #include<queue>
    #include<cstring>
    #define N 210
    #define inf 1000000000000000LL
    using namespace std;
    int n,S,T,num=1,tot,totx,toty,ans,inq[N],f[32005];
    int a[N],b[N],head[N],fa[N],prime[32005],fx[N],fy[N];
    long long c[N],dis[N];
    struct node{
        int from,to,pre,cap;
        long long c;
    }e[N*N];
    bool judge(int x,int y){
        if(!x||!y)return false;
        if(x<y)swap(x,y);
        if(x%y!=0)return 0;
        x/=y;
        for(int i=1;i<=tot;i++){
            if(prime[i]>=x)break;
            if(x%prime[i]==0)return 0;
        }
        return 1;
    }
    void Insert(int from,int to,int v,long long c){
        e[++num].to=to;e[num].from=from;e[num].cap=v;e[num].c=-c;e[num].pre=head[from];head[from]=num;
        e[++num].to=from;e[num].from=to;e[num].cap=0;e[num].c=c;e[num].pre=head[to];head[to]=num;
    }
    bool spfa(){
        queue<int>q;
        memset(inq,0,sizeof(inq));
        for(int i=S;i<=T;i++)dis[i]=inf;
        dis[S]=0;inq[S]=1;q.push(S);
        while(!q.empty()){
            int now=q.front();q.pop();inq[now]=0;
            for(int i=head[now];i;i=e[i].pre){
                int to=e[i].to;
                if(e[i].cap&&dis[to]>dis[now]+e[i].c){
                    dis[to]=dis[now]+e[i].c;fa[to]=i;
                    if(!inq[to]){
                        q.push(to);inq[to]=1;
                    }
                }
            }
        }
        return dis[T]!=inf;
    }
    void min_cost(){
        long long cost=0;
        while(spfa()){
            int tmp=1e9;
            for(int i=fa[T];i;i=fa[e[i].from])
                tmp=min(tmp,e[i].cap);
            if(cost+dis[T]*tmp<=0){
                cost+=dis[T]*tmp;ans+=tmp;
                for(int i=fa[T];i;i=fa[e[i].from])
                    e[i].cap-=tmp,e[i^1].cap+=tmp;
            }
            else {
                ans-=(cost/dis[T]);
                return;
            }
        }
    }
    int main(){
        freopen("menci_pair.in","r",stdin);freopen("menci_pair.out","w",stdout);
        scanf("%d",&n);
        S=0;T=n+1;
        for(int i=1;i<=n;i++)scanf("%d",&a[i]);
        for(int i=1;i<=n;i++)scanf("%d",&b[i]);
        for(int i=1;i<=n;i++)scanf("%lld",&c[i]);
        for(int i=2;i<=32000;i++){
            if(!f[i])prime[++tot]=i;
            for(int j=1;j<=tot;j++){
                if(i*prime[j]>32000)break;
                f[i*prime[j]]=1;
                if(i%prime[j]==0)break;
            }
        }
        for(int i=1;i<=n;i++){
            int tmp=a[i],num=0;
            for(int j=1;j<=tot;j++){
                while(tmp%prime[j]==0)tmp/=prime[j],num++;
                if(tmp==1)break;
            }
            if(num&1)fx[++totx]=i;
            else fy[++toty]=i;
        }
        for(int i=1;i<=totx;i++)
            for(int j=1;j<=toty;j++){
                if(judge(a[fx[i]],a[fy[j]]))
                Insert(fx[i],fy[j],1e9,c[fx[i]]*c[fy[j]]);
            }
        for(int i=1;i<=totx;i++)Insert(S,fx[i],b[fx[i]],0);
        for(int i=1;i<=toty;i++)Insert(fy[i],T,b[fy[i]],0);
        min_cost();
        printf("%d",ans);
        return 0;
    }
    100分 最小费用最大流
  • 相关阅读:
    java 和.net 开发平台的感受(菜鸟级)
    结构体应用统计学生成绩
    实验十 链表
    绘制抛物线(带比例缩放)
    上下三角矩阵的输出
    结构体应用分类与索引
    笔试题之数据库
    动态规划求回文给定字符串,插入字符形成回文
    三行九个点,用4条线段连接(扩展,用3条,用1条)
    名言记录
  • 原文地址:https://www.cnblogs.com/thmyl/p/8058959.html
Copyright © 2011-2022 走看看