博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ3009 : 集合
阅读量:7103 次
发布时间:2019-06-28

本文共 2209 字,大约阅读时间需要 7 分钟。

取一棵生成森林,根据题目限制可得,与一个点相连的多余的边数是$O(\sqrt{m})$级别的。

对于树边,每个点维护3棵权值线段树,依次保存它的儿子里各个集合的边。

再开3*3个分块数组,记录多余边以及树边每种权值的出现次数,修改时暴力修改多余边,时间复杂度$O(q\sqrt{m})$。

 

#include
#include
using namespace std;const int N=100010,M=500010,P=2000000;char op[9];int n,m,Q,i,j,x,y,X,Y,f[N],c[N],ans;int g[N],ge[N],nxt[M<<1],v[M<<1],w[M<<1],ed,q[M][2],cnt,st[N],en[N];int fe[N],T[N][3],a[N][3];bool vis[N];int tot,l[P],r[P],val[P];struct E{int x,y,w;}e[M],b[M];inline bool cmp(const E&a,const E&b){ if(a.x==b.x)return a.y==b.y?a.w
='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}int F(int x){return f[x]==x?x:f[x]=F(f[x]);}inline void add(int x,int y,int z){ v[++ed]=y;w[ed]=z;nxt[ed]=g[x];g[x]=ed; v[++ed]=x;w[ed]=z;nxt[ed]=g[y];g[y]=ed;}inline void adde(int x,int y,int z){ v[++ed]=y;w[ed]=z;nxt[ed]=ge[x];ge[x]=ed; v[++ed]=x;w[ed]=z;nxt[ed]=ge[y];ge[y]=ed;}struct B{ bool v[M];int s[500]; inline void add(int x){v[x]=1,s[x>>10]++;} inline void del(int x){v[x]=0,s[x>>10]--;} inline int ask(){ for(int i=0;i<=m>>10;i++)if(s[i])for(int j=i<<10;;j++)if(v[j])return j; return m+1; }}V[3][3];void ins(int&x,int a,int b,int c,int p){ if(!x)x=++tot;val[x]+=p; if(a==b)return; int mid=(a+b)>>1; if(c<=mid)ins(l[x],a,mid,c,p);else ins(r[x],mid+1,b,c,p);}inline int ask(int x){ if(!val[x])return m+1; int a=1,b=m,mid; while(a
>1; if(val[l[x]])x=l[x],b=mid;else x=r[x],a=mid+1; } return a;}void dfs(int x,int y,int z){ vis[x]=1,f[x]=y,fe[x]=z; for(int i=g[x];i;i=nxt[i])if(!vis[v[i]]){ ins(T[x][0],1,m,w[i],1); dfs(v[i],x,w[i]); } for(int i=0;i<3;i++)V[0][i].add(a[x][i]=ask(T[x][i]));}int main(){ read(n),read(m); for(i=1;i<=n;i++)f[i]=i; for(i=1;i<=m;i++){ read(e[i].x),read(e[i].y),read(e[i].w); if(e[i].x>e[i].y)swap(e[i].x,e[i].y); } sort(e+1,e+m+1,cmp); for(i=1;i<=m;i++)if(e[i].x!=e[i-1].x||e[i].y!=e[i-1].y)b[++j]=e[i]; for(m=j,sort(b+1,b+m+1,cmp2),i=1;i<=m;i++){ x=b[i].x,y=b[i].y; if(F(x)!=F(y))f[f[x]]=f[y],add(x,y,i);else adde(x,y,i); } for(i=1;i<=n;en[i++]=cnt){ if(!vis[i])dfs(i,0,0); for(st[i]=cnt+1,j=ge[i];j;j=nxt[j]){ if(v[j]

  

转载地址:http://ugdhl.baihongyu.com/

你可能感兴趣的文章
我的友情链接
查看>>
大数据处理相关的好博文
查看>>
essential C++
查看>>
Git 服务器搭建与客户端安装
查看>>
使用 Java8 Optional 的正确姿势
查看>>
[C++ 学习笔记 1] delete 和 delete [] 的本质区别
查看>>
quartz 2.0.2 hello
查看>>
关于编程工具链
查看>>
Android新的ARM开发工具包 解决平台混乱问题
查看>>
TensorFlow人工智能引擎入门教程之二 CNN卷积神经网络的基本定义理解。
查看>>
Linux系统新手学习的11点建议
查看>>
Github上传代码菜鸟超详细教程【转】
查看>>
SVN上的项目如何迁移到Git
查看>>
多级<select>选择的实现(利用selectedIndex属性)
查看>>
Apache Rewrite
查看>>
转贴: QUARTUS 实现远程控制的简单方法
查看>>
开源还是商用?十大云运维监控工具横评
查看>>
python3 科学计算2
查看>>
Mysql启动失败Can’t connect to local MySQL server throu
查看>>
大学四年的学习经历
查看>>