1. 程式人生 > >Cogs 2221. [SDOI2016 Round1] 數字配對

Cogs 2221. [SDOI2016 Round1] 數字配對

輸出格式 queue prim bsp 價值 pos rac 因數分解 stream

2221. [SDOI2016 Round1] 數字配對

題目鏈接

★★★ 輸入文件:menci_pair.in 輸出文件:menci_pair.out 簡單對比
時間限制:1 s 內存限制:128 MB

【題目描述】

n 種數字,第 i 種數字是 ai、有 bi 個,權值是 ci

若兩個數字 aiaj 滿足,aiaj 的倍數,且 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:n200

ai109bi105ci∣≤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分 最小費用最大流

Cogs 2221. [SDOI2016 Round1] 數字配對