【poj1201】 Intervals

http://poj.org/problem?id=1201 (題目鏈接)

題意:給出n個區間[ai,bi],要求選出儘可能少的數,使得每個區間i中至少存在c[i]個數。

Solution
  差分約束。
  區間可以表示爲sum[bi]-sum[ai-1],所以可以列出n個不等式:sum[bi]-sum[ai-1]>=c[i],然後每個sum[x]滿足0<=sum[x+1]-sum[x]<=1 ——> sum[x+1]-sum[x]>=0,sum[x]-sum[x+1]>=-1。這樣的話構圖就確定了連通性。建完圖後,跑SPFA最長路即可。

代碼:

// poj1201
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#define LL long long
#define inf 2147483640
#define MOD 998244353
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std;
inline LL getint() {
    int f,x=0;char ch=getchar();
    while (ch<='0' || ch>'9') {if (ch=='-') f=-1;else f=1;ch=getchar();}
    while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
    return x*f;
}

const int maxn=50010;
struct edge {int to,w,next;}e[maxn<<2];
int vis[maxn],dis[maxn],head[maxn],n,cnt,L,R;

void insert(int u,int v,int w) {
    e[++cnt].to=v;e[cnt].next=head[u];head[u]=cnt;e[cnt].w=w;
}
int SPFA() {
    queue<int> q;
    for (int i=L;i<=R;i++) {
        vis[i]=1;
        dis[i]=0;
        q.push(i);
    }
    while (q.size()) {
        int x=q.front();
        q.pop();
        vis[x]=0;
        for (int i=head[x];i;i=e[i].next)
            if (e[i].w+dis[x]>dis[e[i].to]) {
                dis[e[i].to]=e[i].w+dis[x];
                if (!vis[e[i].to]) {vis[e[i].to]=1;q.push(e[i].to);}
            }
    }
    return dis[R];
}
int main() {
    while (scanf("%d",&n)!=EOF) {
        memset(head,0,sizeof(head));
        L=inf,R=0;
        for (int u,v,w,i=1;i<=n;i++) {
            scanf("%d%d%d",&u,&v,&w);
            insert(u-1,v,w);
            L=min(L,u-1);
            R=max(R,v);
        }
        for (int i=L;i<=R;i++) {
            insert(i,i+1,0);
            insert(i+1,i,-1);
        }
        printf("%d\n",SPFA());
    }
    return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章