• 【BZOJ3218】 a+b Problem



    Solution

      看一下如何建最小割模型:

      先求$sum=sum b_i+w_i$。

      如果某一个格子$i$选择了黑色,那么贡献是$-w_i$;如果满足是它奇怪的格子,还有额外贡献$-p_i$。

      如果选择了白色,那么贡献是$-b_i$。

      总贡献加上$sum$就是答案。

      对于如上条件,可以对每一个格子建立一个单元:

        

      可以发现,如果选择白色,那么直接割去$b[i]$这条边。那么要怎么影响之后的黑格呢?

      对于每一个点$i$,对于所有的$j>i$,使得$i$按题目要求可以使$j$成为奇怪的格子,那么由$i$向$j'$连一条$infty$的边。

      如果第$i$个格子选择白色,那么对$j$的单元来说,必须割掉$p[j]$这条边,加上原本一定要割的$w[j]$,就是$j$成为奇怪格子的贡献。 

      由此建模完成。

      但是边数高达$O(n^2)$

      看看题目的要求是$i<j$,也就是说对于格子$i$,在$1...i-1$的范围内找权值为$[l_i,r_i]$的格子,对单元进行连接。

      看起来好像主席树啊,那就关于$a_i$建权值主席树吧。

      类似线段树优化网络流建边地,往主席树插入第$i$个格子的时候,将$i$(单元里面的那个$i$点)连向新链上每一个新节点,再由原节点连向新节点,流量都是$infty$。

      注意不能像平时用线段树跑网络流一样在叶子底部连接,然后从儿子往父亲流。因为这样会出现版本的冲突。

      跑一下就好了。

      PS:边的计数器忘记初始化调了整整一个上午服了。。。

      


    #include <cstdio>
    #include <algorithm>
    #include <queue>
    using namespace std;
    const int N=5010,NS=100010,INF=2147000000;
    int n,a[N],b[N],w[N],lb[N],rb[N],p[N],sum;
    int h[NS],tot,cnt,root[N];
    int list[NS],ltot,maxs;
    int ch[NS][2];
    int S,T,dis[NS],cur[NS];
    queue<int> q;
    struct Edge{int v,f,next;}g[1000010];
    inline void addEdge(int u,int v,int f){
        g[++tot].v=v; g[tot].f=f; g[tot].next=h[u]; h[u]=tot;
        g[++tot].v=u; g[tot].f=0; g[tot].next=h[v]; h[v]=tot;
    }
    void lsh(){
        sort(list+1,list+1+ltot);
        ltot=unique(list+1,list+1+ltot)-list-1;
        maxs=ltot;
        for(int i=1;i<=n;i++){
            a[i]=lower_bound(list+1,list+1+ltot,a[i])-list;
            lb[i]=lower_bound(list+1,list+1+ltot,lb[i])-list;
            rb[i]=lower_bound(list+1,list+1+ltot,rb[i])-list;
        }
    }
    int copy(int u){
        int v=++cnt;
        ch[v][0]=ch[u][0]; ch[v][1]=ch[u][1];
        if(u)
            addEdge(u,v,INF);
        return v;
    }
    void insert(int u,int &v,int l,int r,int pos,int who){
        v=copy(u);        
        addEdge(who*2-1,v,INF);
        if(l==r) return;
        int mid=(l+r)>>1;
        if(pos<=mid)
            insert(ch[u][0],ch[v][0],l,mid,pos,who);
        else
            insert(ch[u][1],ch[v][1],mid+1,r,pos,who);
    }
    void link(int u,int l,int r,int L,int R,int node){
        if(!u) return;
        if(L<=l&&r<=R){
            addEdge(u,node,INF);
            return;
        }
        int mid=(l+r)>>1;
        if(R<=mid) link(ch[u][0],l,mid,L,R,node);
        else if(mid<L) link(ch[u][1],mid+1,r,L,R,node);
        else{
            link(ch[u][0],l,mid,L,mid,node);
            link(ch[u][1],mid+1,r,mid+1,R,node);
        }
    }
    bool bfs(){
        while(!q.empty()) q.pop();
        q.push(S);
        for(int i=1;i<=cnt;i++) dis[i]=-1;
        dis[S]=0;
        while(!q.empty()){
            int u=q.front(); q.pop();
            for(int i=h[u],v;i;i=g[i].next)
                if(g[i].f&&dis[v=g[i].v]==-1){
                    dis[v]=dis[u]+1;
                    if(v==T) return true;
                    q.push(v);
                }
        }
        return dis[T]!=-1;
    }
    int dfs(int u,int delta){
        if(u==T) return delta;
        int ret=0,get;
        for(int i=cur[u],v;i&&delta;i=g[i].next)
            if(g[i].f&&dis[v=g[i].v]==dis[u]+1){
                get=dfs(v,min(delta,g[i].f));
                g[i].f-=get;
                g[i^1].f+=get;
                if(g[i].f) cur[u]=i;
                delta-=get;
                ret+=get;
            }
        if(!ret) dis[u]=-1;
        return ret;
    }
    int dinic(){
        int ret=0;
        while(bfs()){
            for(int i=1;i<=cnt;i++) cur[i]=h[i];
            ret+=dfs(S,INF);
        }
        return ret;
    }
    int main(){
        scanf("%d",&n);
        for(int i=1;i<=n;i++){
            scanf("%d%d%d%d%d%d",&a[i],&b[i],&w[i],&lb[i],&rb[i],&p[i]);
            list[++ltot]=a[i];
            list[++ltot]=lb[i];
            list[++ltot]=rb[i];
            sum+=b[i]+w[i];
        }
        lsh();
        S=n*2+1; T=n*2+2;
        tot=1;
        for(int i=1;i<=n;i++){
            cnt+=2;
            addEdge(cnt,cnt-1,p[i]);
            addEdge(S,cnt-1,w[i]);
            addEdge(cnt-1,T,b[i]);
        }
        cnt+=2;
        for(int i=1;i<=n;i++)
            insert(root[i-1],root[i],1,maxs,a[i],i);    
        for(int i=2;i<=n;i++)
            link(root[i-1],1,maxs,lb[i],rb[i],i*2);
        int get=dinic();
        printf("%d
    ",sum-get);
        return 0;
    }
    奇妙代码
  • 相关阅读:
    VC++以及VS个版本比较 及 C++编译器比较
    这本书的封面
    json_encode(),json_deocde()用法说明
    循环匹配,一般用于多选列表选中状态
    js判断变量是否被定义
    js onload()事件调用方法
    js push使用方法
    iframe去掉边滚动条
    ajax上传文件(使用ajaxfileupload)
    基于url的权限控制
  • 原文地址:https://www.cnblogs.com/RogerDTZ/p/8016382.html
Copyright © 2020-2023  润新知