首页 > 其他分享 >P5163 WD与地图

P5163 WD与地图

时间:2023-12-27 11:34:49浏览次数:37  
标签:WD bb int 地图 SCC 200010 P5163 st now

更好的阅读体验

P5163 WD与地图

喵喵题,但其实没有那么难。

删边倒序转成加边是显然的,询问可以通过值域线段树合并实现,修改,合并,查询都是好做的。考虑如何维护动态加边的 SCC。

难点是每个时刻缩点后的图是一个 DAG,并不像无向图的搜索树一样好维护,而且新加入的边可能不会立刻构成 SCC 而是和再之后加入的边构成。

简化问题,可以通过计算出每条边被并到某个 SCC 里的时刻来维护。发现每条边只会被并到某个 SCC 里一次(之后的合并是边所在的 SCC 和其他的 SCC 合并,和这条边无关),满足单调性。并且需要对于每条边都求出这个时间,考虑整体二分。

简单的想法是,对于当前的二分区间,加入 \(l\sim mid\) 间的新边和这个区间询问(询问是指对一条边什么时候合并的询问)的所有边,跑一遍 Tarjan 求出 SCC,对于所有询问边,已经属于一个 SCC 的全部向左侧递归。对于在右侧的,先加入左侧的所有边,然后向右侧递归。

但是上面的复杂度是假的,原因很显然,每次 Tarjan 的复杂度和 \(r\) 有关而不是和 \(r-l+1\) 有关,其本质是没有利用好 SCC 的性质,加入了大量的无用边导致复杂度退化。

把上面的做法稍作改进:跑完 Tarjan 之后,把 SCC 内所有点用并查集合并在一起,然后向右侧递归。加边的时候加入 \((find(x)-find(y))\) 这样每次跑 Tarjan 的时候复杂度就挂在了边数上。然后向左侧递归的时候需要把并查集取消,可以用可撤销并查集实现,总复杂度 \(\mathcal O(n\log^2n)\)。(另一种实现方式是把右侧边进行重标号,可以消去并查集的一只 \(\log\),本人采用并查集写法)

注意实现细节:询问边有一部分就是新边,二分的在某个区间一个已经被删掉的询问边可能还没被加进来;注意特判开始就在 SCC 里的边和最后也没被合并的边。细节挺多的,写了一上午,但是一遍过?离谱。

也算是比较重工业的题了,代码写得很丑,见谅/wul。

	int n,m,q,len,all,vis[200010],ans[200010],a[200010],numa[400010];
	pii bb[200010];
	tup b[200010],c[200010],le[200010],ri[200010],d[200010];
	map<pii,int> hash;
	vector<pii> edge;
	vector<int> ver,del[200010],ve[200010];
	#define id(x,y) (x-1)*100000+y
	namespace RDSU
	{
		int fa[200010],siz[200010];
		stack<int> st;
		inline void init(){for(int i=1;i<=n;++i)siz[i]=1,fa[i]=i;}
		inline int find(int x){while(x!=fa[x])x=fa[x];return x;}
		inline bool Dmerge(int x,int y)
		{
			if((x=find(x))==(y=find(y)))return 0;
			if(siz[x]>siz[y])st.e(y),siz[x]+=siz[y],fa[y]=x;
			else st.e(x),siz[y]+=siz[x],fa[x]=y;
			return 1;
		}
		inline void cancel(){int x=st.top();st.pop();siz[fa[x]]-=siz[x],fa[x]=x;}
	}
	using namespace RDSU;
	namespace Segment
	{
		int root[200010],cnt;
		struct{int ls,rs,val;ll sum;}t[16000010];
		inline void update(int now)
		{
			t[now].val=t[t[now].ls].val+t[t[now].rs].val;
			t[now].sum=t[t[now].ls].sum+t[t[now].rs].sum;
		}
		void change(int&now,int x,int y,int L=0,int R=len)
		{
			if(!now)now=++cnt;
			t[now].val+=numa[x]*y,t[now].sum+=y;
			if(L==R)return void();
			int mid=L+((R-L)>>1);
			if(x<=mid)change(t[now].ls,x,y,L,mid);
			else change(t[now].rs,x,y,mid+1,R);
		}
		int merge(int x,int y,int L=0,int R=len)
		{
			if(!x||!y)return x|y;
			if(L==R)return t[x].val+=t[y].val,t[x].sum+=t[y].sum,x;
			int mid=L+((R-L)>>1);
			t[x].ls=merge(t[x].ls,t[y].ls,L,mid);
			t[x].rs=merge(t[x].rs,t[y].rs,mid+1,R);
			return update(x),x;
		}
		int ask(int now,int x,int L=0,int R=len)
		{
			if(x>=t[now].sum)return t[now].val;
			if(!x)return 0;
			if(L==R)return numa[L]*x;
			int mid=L+((R-L)>>1);
			if(x<=t[t[now].rs].sum)return ask(t[now].rs,x,mid+1,R);
			return ask(t[now].rs,x,mid+1,R)+ask(t[now].ls,x-t[t[now].rs].sum,L,mid);
		}
		void print(int p,int L=0,int R=len)
		{
			if(!p)return;
			if(L==R)return write('(',L,',',t[p].sum,')');
			int mid=L+((R-L)>>1);
			print(t[p].ls,L,mid),print(t[p].rs,mid+1,R);
		}
	}
	using namespace Segment;
	namespace Connection
	{
		int dfn[200010],low[200010],tot,num,col[200010],ins[200010];
		stack<int> st;
		void tarjan(int x)
		{
			st.e(x),dfn[x]=low[x]=++tot,ins[x]=1;
			for(auto to:ve[x])
			{
				if(!dfn[to])tarjan(to),low[x]=min(low[x],low[to]);
				else if(ins[to])low[x]=min(low[x],dfn[to]);
			}
			if(dfn[x]==low[x])
			{
				int y;++num;
				do ins[y=st.top()]=0,col[y]=num,st.pop();while(y!=x);
			}
		}
		
	}
	using namespace Connection;
	bitset<200010> viss;
	void solve(int l,int r,int L,int R)
	{
		if(L==R)
		{
			for(int i=l;i<=r;++i)
			if(vis[b[i].z]!=inf)
			del[L].eb(b[i].z);
			return;
		}
		if(l>r)return;
		int mid=L+((R-L)>>1),len1=0,len2=0;
		for(int i=L,x,y;i<=mid;++i)
		{
			if(vis[d[i].z]==inf)continue;
			x=find(d[i].x),y=find(d[i].y);
			if(!viss[x])viss[x]=1,ver.eb(x);
			if(!viss[y])viss[y]=1,ver.eb(y);
			ve[x].eb(y);
		}
		for(int i=l;i<=r;++i)
		{
			if(vis[b[i].z]>mid)continue;
			bb[i].fi=find(b[i].x),bb[i].se=find(b[i].y);
			if(!viss[bb[i].fi])viss[bb[i].fi]=1,ver.eb(bb[i].fi);
			if(!viss[bb[i].se])viss[bb[i].se]=1,ver.eb(bb[i].se);
			ve[bb[i].fi].eb(bb[i].se);
		}
		for(auto j:ver)if(!dfn[j])tarjan(j);
		vector<pii> ved;
		for(int i=l;i<=r;++i)
		{
			if(vis[b[i].z]<=mid&&col[bb[i].fi]==col[bb[i].se])
			ved.eb(bb[i].fi,bb[i].se),le[++len1]=b[i];
			else ri[++len2]=b[i];
		}
		for(auto j:ver)low[j]=dfn[j]=col[j]=0,ve[j].clear();
		ver.clear(),tot=num=0;
		for(int i=L;i<=mid;++i)viss[find(d[i].x)]=viss[find(d[i].y)]=0;
		for(int i=l;i<=r;++i)viss[bb[i].fi]=viss[bb[i].se]=0;
		for(int i=1;i<=len1;++i)b[l+i-1]=le[i];
		for(int i=1;i<=len2;++i)b[l+len1+i-1]=ri[i];
		solve(l,l+len1-1,L,mid);
		for(auto p:ved)Dmerge(p.fi,p.se);
		solve(l+len1,r,mid+1,R);
	}
	inline int vfind(int x){return lower_bound(numa+1,numa+1+len,x)-numa;}
	inline void Merge(int x,int y){if((x=find(x))!=(y=find(y)))root[x]=root[y]=merge(root[x],root[y]),Dmerge(x,y);}
	inline bool cmp(tup t1,tup t2){return t1.z<t2.z;}
	inline void mian()
	{
		read(n,m,q),init();
		for(int i=1;i<=n;++i)read(a[i]),numa[++len]=a[i];
		for(int i=1;i<=m;++i)read(b[i].x,b[i].y),hash[mp(b[i].x,b[i].y)]=b[i].z=i;
		for(int i=1;i<=q;++i)
		{
			read(c[i].x,c[i].y,c[i].z);
			if(c[i].x==1)d[++all]=b[hash[mp(c[i].y,c[i].z)]],vis[hash[mp(c[i].y,c[i].z)]]=all;
			if(c[i].x==2)numa[++len]=(a[c[i].y]+=c[i].z);
		}
		for(int i=1;i<=m;++i)if(vis[i])vis[i]=all-vis[i]+1;
		for(int i=1;i<=m;++i)ve[b[i].x].eb(b[i].y);
		for(int i=1;i<=n;++i)if(!dfn[i])tarjan(i);
		for(int i=1;i<=m;++i)if(col[b[i].x]!=col[b[i].y])vis[i]=inf;
		tot=num=0;
		for(int i=1;i<=n;++i)dfn[i]=low[i]=0,ve[i].clear(),col[i]=0;
		sort(numa+1,numa+1+len),len=unique(numa+1,numa+1+len)-numa-1;
		reverse(d+1,d+1+all),reverse(c+1,c+1+q),init();
		if(all)solve(1,m,1,all),sort(b+1,b+1+m,cmp);
		for(int i=1;i<=n;++i)change(root[i],vfind(a[i]),1);
		for(int i=1;i<=m;++i)if(!vis[i])ve[b[i].x].eb(b[i].y);
		for(int i=1;i<=n;++i)if(!dfn[i])tarjan(i);
		for(int i=1;i<=m;++i)if(!vis[i]&&col[b[i].x]==col[b[i].y])Merge(b[i].x,b[i].y);
		for(int i=1,t=0,x,y;i<=q;++i)
		{
			if(c[i].x==1)
			{
				++t;
				for(auto j:del[t])
				{
					if((x=find(b[j].x))!=(y=find(b[j].y)))
					Merge(x,y);
				}
			}
			else if(c[i].x==2)
			{
				change(root[find(c[i].y)],vfind(a[c[i].y]),-1);
				change(root[find(c[i].y)],vfind(a[c[i].y]-=c[i].z),1);
			}
			else ans[i]=ask(root[find(c[i].y)],c[i].z);
		}
		for(int i=q;i>=1;--i)if(c[i].x==3)write(ans[i],'\n');

标签:WD,bb,int,地图,SCC,200010,P5163,st,now
From: https://www.cnblogs.com/WrongAnswer90-home/p/17930208.html

相关文章

  • Leaflet 使用图片作为地图
    Leaflet使用图片作为地图关键代码:L.CRS.Simple.transformation=newL.Transformation(1,0,1,0);//坐标原点切换为左上角varmap=newL.Map('map',{crs:L.CRS.Simple,//设置地图坐标模式为简单模式center:[0,0],//地图中心zoom:-0.5,//缩放......
  • 融云获评「全球领航者·年度服务商」,自制《地图》引领行业风潮
    12月19日,由新黄河、经济观察报与霞光智库共同举办的“潮起·奔流——2023全球领航者大会”在北京举办。关注【融云全球互联网通信云】了解更多大会重磅发布“全球领航者2023年度榜单”,融云获评“全球领航者·年度服务商”。作为在出海大年收尾时举办的一场总结大会,众多出海......
  • 使用 Flutter 制作地图应用
    使用Flutter制作地图应用本文主要介绍使用Flutter制作地图应用在本文中,我将向您展示如何使用Flutter向您的应用程序添加映射功能。对于本教程,您将不需要googlemapsAPI,因此您无需支付任何费用,因为我们将使用另一个免费API,所以不用多说,让我们深入研究它。依赖关系创建一个......
  • 3D组合地图在数据可视化大屏中的应用
    前言当下数据可视化大屏展示的花样层出不穷,可视化大屏的C位越来越卷,地图的样式已经不再止步于普通的平面地图,在虚拟环境中探索和交互,今天我们要介绍的这一款3D组合地图可以将复杂的数据以直观的方式呈现出来,使得数据更容易被理解和分析。例如,通过将人口分布、经济状况等数据与3D......
  • 地图服务器GeoServer的安装与配置
    目录1.安装配置Java2安装配置Tomcat3安装配置GeoServerGeoServer提供了多种安装配置方式,但是本质上GeoServer是一个基于JavaWeb的项目,因此我们理论上只需要安装Java,并且将其放置在一个Web服务器(例如ApacheTomcat)下进行发布就可以了。另外,GeoServer还提供了包含ApacheTomcat......
  • echarts在vue中不显示中国地图
    遇到的情况是在开发中,中国地图是正常显示的但是打包之后,放到服务器,地图就不显示了,经过搜索,说是因为低版本的echarts需要手动添加map首先找到这个文件然后手动写上这个最后重新打包试试吧......
  • 通过tidevice 启动wda 提示: request error: ('Connection aborted.', MuxReplyError(
    当我在使用tidevice启动wda来做iOS自动化测试的时候一直会报错:requesterror:('Connectionaborted.',MuxReplyError(<UsbmuxReplyCode.ConnectionRefused:3>))我在网上也一直翻翻翻寻找答案,每一次眼看着就快解决了可到最后仍是出现这串错误❌,经过几番波折我能试的办法都试了......
  • blazor maui hybrid app显示本地图片
    啊......一通操作下来感觉就是两个字折磨跨平台有跨平台的好处但框架本身支持的有限很多东西做起来很曲折哎这里总结一下笔者为了折腾本地图片显示的尝试为什么要做本地图片展示呢如果是做需要网络连接的app这个一般是不需要的(要做上传前预览/编辑的话还是要的)但对......
  • 在Linux环境下模拟实现命令解释器用c语言实现mypwd「粉丝答疑」
    Solution要在Linux环境下用C语言模拟实现一个命令解释器,包含mypwd,mymkdir,myrmdir,mycd,mylist,mycp,mydate,mycreate,mydelete,exit等基本命令,需要按照以下步骤进行:理解每个命令的功能:mypwd:显示当前工作目录。mymkdir:创建一个新目录。myrmdir:删除一个空目......
  • 地图坐标转换 WGS84、BD09与GCJ02的相互转换
    高德地图WGS84转GCJ02exportfunctionwgs84ToGcj02(lng,lat){if(out_of_china(lng,lat)){return[lng,lat]}else{vardlat=transformlat(lng-105.0,lat-35.0)vardlng=transformlng(lng-105.0,lat-35.0)......