hdu 1384 差分约束(SPFA实现)

Problem: http://acm.hdu.edu.cn/showproblem.php?pid=1384

在每个区间[ai,bi]上至少选ci个元素组成集合Z,求Z最小时的元素个数

因为是求最小,所以差分约束转化为SPFA求最长路

每个ai,bi,ci转换成点ai指向点bi+1的边,边权为ci

求出最小的ai,Min,最大的bi,Max

加上(Max-Min+2)*2条有向边x->x+1(边权为0)x+1->x(边权为-1) x属于[Min,Max]

最后套上SPFA模版妥妥的

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
#define MAXN 100010
int head[MAXN],n,m,Min,Max,dist[MAXN];
struct Edge{
    int to,next,w;
}e[MAXN*3];
void add_edge(int u,int v,int c){
    e[m].to=v;
    e[m].w=c;
    e[m].next=head[u];
    head[u]=m++;
}
void SPFA(int s){
    bool vis[MAXN];
    memset(vis,false,sizeof(vis));
    for(int i=Min;i<=Max+1;i++)dist[i]=-10000;
    queue<int> que;
    que.push(s);
    vis[s]=false;
    dist[s]=0;
    while(!que.empty()){
        int u=que.front();
        que.pop();
        vis[u]=false;
        int k=head[u];
        while(k>=0){
            if(dist[e[k].to]<dist[u]+e[k].w){
                dist[e[k].to]=dist[u]+e[k].w;
                if(!vis[e[k].to]){
                    que.push(e[k].to);
                    vis[e[k].to]=true;
                }
            }
            k=e[k].next;
        }
    }
}
void init(){
    memset(head,-1,sizeof(head));
    int a,b,c;
    Max=m=0;
    Min=MAXN;
    for(int i=0;i<n;i++){
        scanf("%d%d%d",&a,&b,&c);
        Min=min(Min,a);
        Max=max(Max,b);
        add_edge(a,b+1,c);
    }
    for(int i=Min;i<=Max;i++){
        add_edge(i,i+1,0);
        add_edge(i+1,i,-1);
    }
}
int main()
{
    while(scanf("%d",&n)!=EOF){
        init();
        SPFA(Min);
        printf("%d\n",dist[Max+1]);
    }
    return 0;
}
View Code

加上set容器优化了一下

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#include<set>
using namespace std;
#define MAXN 100010
int head[MAXN],n,m,Min,Max,dist[MAXN];
struct Edge{
    int to,next,w;
}e[MAXN*3];
void add_edge(int u,int v,int c){
    e[m].to=v;
    e[m].w=c;
    e[m].next=head[u];
    head[u]=m++;
}
void SPFA(int s){
    bool vis[MAXN];
    memset(vis,false,sizeof(vis));
    for(int i=Min;i<=Max+1;i++)dist[i]=-10000;
    queue<int> que;
    que.push(s);
    vis[s]=false;
    dist[s]=0;
    while(!que.empty()){
        int u=que.front();
        que.pop();
        vis[u]=false;
        int k=head[u];
        while(k>=0){
            if(dist[e[k].to]<dist[u]+e[k].w){
                dist[e[k].to]=dist[u]+e[k].w;
                if(!vis[e[k].to]){
                    que.push(e[k].to);
                    vis[e[k].to]=true;
                }
            }
            k=e[k].next;
        }
    }
}
void init(){
    memset(head,-1,sizeof(head));
    int a,b,c;
    Max=m=0;
    Min=MAXN;
    set<int> S;
    for(int i=0;i<n;i++){
        scanf("%d%d%d",&a,&b,&c);
        Min=min(Min,a);
        Max=max(Max,b);
        add_edge(a,b+1,c);
        S.insert(a);
        S.insert(b+1);
    }
    b=MAXN;
    set<int>::iterator it=S.end();
    while(it!=S.begin()){//减少边来降低计算量
        --it;
        a=*it;
        add_edge(a,b,0);
        add_edge(b,a,a-b);
        b=a;
    }
}
int main()
{
    while(scanf("%d",&n)!=EOF){
        init();
        SPFA(Min);
        printf("%d\n",dist[Max+1]);
    }
    return 0;
}
View Code

话说200+ms是怎么做到的...

posted @ 2014-02-15 01:15  Cshhr  阅读(296)  评论(0编辑  收藏  举报