zoukankan      html  css  js  c++  java
  • [HNOI2010]城市建设

    IV.II.[HNOI2010]城市建设

    实际上这题不算狭义上的CDQ分治(先计算左边,再计算左边对右边的贡献,最后计算右边),更像是线段树分治的变种,但是既然大家都认为这就是CDQ那就算是罢……


    考虑分治计算。当我们考虑一个区间 \([l,r]\) 时,我们会将所有边分为两类:区间 \([l,r]\) 内操作涉及到的边,称作动态边;未涉及到的边,称作静态边。显然,对于 \([l,r]\) 的子集来说,静态边只会增加不会减少,因此我们考虑在 \([l,r]\) 处就着手缩减静态边集中不需要的部分。

    我们发现,对于 \([l,r]\) 中的动态边来说,就算它们全连上也不一定会使得整张图联通,因而连上所有动态边后剩余的连通块间就必然要使用静态边连接。于是这部分的静态边就可以直接跑个MST来联通这些剩余的连通块。显然,此时MST上连出的边,在子区间里是一定会被保留的,故可以缩点。

    于是总结一下此部分的操作:

    1. 对动态边缩点

    2. 对静态边求MST

    3. 撤销上述所有操作直到1执行之前,然后连回2中求出的MST上的边。(明显,此部分要使用可撤销冰茶姬)

    于是我们现在便率先压缩了点数,并删除了部分静态边。

    在上述操作之后,我们又发现一点可乘之机:有部分静态边是不可能出现在最终的MST上的,因为此时它们已经可以被其它静态边替代掉了。

    于是我们要再跑一遍静态边的MST,并丢弃掉所有不在MST上的静态边。

    显然,一个区间中动态边数量是区间长度级别的;故第一次压缩点数后所剩余的点数也是区间长度级别的;故第二次压缩边数后所剩余的边数也是区间长度级别的;于是我们发现经历上述操作后,区间动态边、静态边数均是区间长度级别的;于是复杂度为分治的一个 \(\log\) 乘以(MST的一个 \(\log\) 以及可撤销冰茶姬的一个 \(\log\)),为 \(n\log^2n\)

    需要注意的是,在递归到区间长度为 \(1\) 时,就直接把修改操作执行,然后暴力跑个MST就行了。

    (还有一种解法是非常trivial的线段树分治套LCT,虽然也是 \(O(n\log^2n)\) 但是有着众所周知的大常数因此跑不过去,虽然如果省选时遇到这道题我肯定就暴力上LCT了,因为CDQ分治的解法太太太恶心了)

    代码:

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    int n,m,q,id[50010],val[50010],dsu[20010],sz[20010];
    struct dat{int u,v,su,sv;dat(int U,int V,int SU,int SV){u=U,v=V,su=SU,sv=SV;}};
    stack<dat>s;
    void undo(){dsu[s.top().u]=s.top().u,dsu[s.top().v]=s.top().v,sz[s.top().u]=s.top().su,sz[s.top().v]=s.top().sv,s.pop();}
    int find(int x){return dsu[x]==x?x:find(dsu[x]);}
    struct EDGE{
    	int x,y,z;
    	bool merge(){
    		int X=find(x),Y=find(y);if(X==Y)return false;
    		if(sz[X]>sz[Y])swap(X,Y);
    		s.push(dat(X,Y,sz[X],sz[Y])),dsu[X]=Y,sz[Y]+=sz[X];
    		return true;
    	}
    }e[50010];
    ll res,ans[50010];
    vector<int>stl,mov;//still edges and movable edges
    bool nst[50010];//if the edge is non-still
    bool cmp(int x,int y){return e[x].z<e[y].z;}
    void solve(int l,int r){
    	ll RES=res;vector<int>STL=stl,MOV=mov;int SZ=s.size();//those are the old information
    	
    //	printf("BEF:[%d,%d]\n",l,r);printf("STL:");for(auto i:stl)printf("%d ",i);puts("");printf("MOV:");for(auto i:mov)printf("%d ",i);puts("");
    	
    	for(int i=l;i<=r;i++)nst[id[i]]=true;
    	vector<int>tmp;
    	for(auto i:mov)if(nst[i])tmp.push_back(i);else stl.push_back(i);//separate those once-movable edges into still-movable edges and newly-still edges
    	for(int i=l;i<=r;i++)nst[id[i]]=false;
    	mov=tmp;
    	
    //	printf("MID:[%d,%d]\n",l,r);printf("STL:");for(auto i:stl)printf("%d ",i);puts("");printf("MOV:");for(auto i:mov)printf("%d ",i);puts("");
    	
    //	for(int i=1;i<=n;i++)printf("%d ",find(i));puts("");
    	int S=s.size();
    	for(auto i:mov)e[i].merge();//merge movable edges, to find those must-add edges 
    	
    	tmp.clear();vector<int>pmt;//pmt holds the must-add edges.
    	sort(stl.begin(),stl.end(),cmp);for(auto i:stl)if(e[i].merge())res+=e[i].z,pmt.push_back(i);else tmp.push_back(i);
    	stl=tmp;while(s.size()>S)undo();
    	for(auto i:pmt)e[i].merge();//now must-add edges have all been merged.
    	
    //	for(int i=1;i<=n;i++)printf("%d ",find(i));puts("");
    	tmp.clear(),S=s.size();
    	for(auto i:stl)if(e[i].merge())tmp.push_back(i);
    	while(s.size()>S)undo();stl=tmp;//now stl holds the minimum-spaning-forest,of those current-still edges
    	
    //	printf("AFT:[%d,%d]\n",l,r);printf("STL:");for(auto i:stl)printf("%d ",i);puts("");printf("MOV:");for(auto i:mov)printf("%d ",i);puts("");
    	
    	if(l==r){
    		e[id[l]].z=val[l],stl.push_back(id[l]);
    		sort(stl.begin(),stl.end(),cmp);for(auto i:stl)if(e[i].merge())res+=e[i].z;
    		ans[l]=res;
    	}else{int mid=(l+r)>>1;solve(l,mid),solve(mid+1,r);}
    	while(s.size()>SZ)undo();res=RES,stl=STL,mov=MOV;
    }
    int main(){
    	scanf("%d%d%d",&n,&m,&q);
    	for(int i=1;i<=n;i++)dsu[i]=i,sz[i]=1;
    	for(int i=1;i<=m;i++)scanf("%d%d%d",&e[i].x,&e[i].y,&e[i].z),mov.push_back(i);//initially all edges are considered movable
    	for(int i=1;i<=q;i++)scanf("%d%d",&id[i],&val[i]);
    	solve(1,q);
    	for(int i=1;i<=q;i++)printf("%lld\n",ans[i]);
    	return 0;
    } 
    

  • 相关阅读:
    老生长谈:css实现右侧固定宽度,左侧宽度自适应
    链接rel属性external、nofollow、external nofollow三种写法的区别
    文字无缝向上滚动
    JS中判断鼠标按键的问题
    js 字符串转换数字
    jS字符串大小写转换实现方式
    JS截取字符串
    CentOS中查看物理CPU信息的方法
    如何开启MYSQL远程连接权限
    PHP中数组排序实例学习
  • 原文地址:https://www.cnblogs.com/Troverld/p/14620777.html
Copyright © 2011-2022 走看看