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

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

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

#include<cstdio>
#include<algorithm>
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<b.w:a.y<b.y;
return a.x<b.x;
}
inline bool cmp2(const E&a,const E&b){return a.w<b.w;}
inline void read(int&a){char c;while(!(((c=getchar())>='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<b){
mid=(a+b)>>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]<i)V[0][0].add(w[j]);
q[++cnt][0]=v[j],q[cnt][1]=w[j];
}
}
read(Q);
while(Q--){
scanf("%s",op);
if(op[0]=='A'){
ans=min(V[op[3]-'A'][op[4]-'A'].ask(),V[op[4]-'A'][op[3]-'A'].ask());
if(ans<=m)printf("%d\n",b[ans].w);else puts("No Found!");
}else{
read(x);y=op[4]-'A';X=c[x];
if(X==y)continue;
if(f[x]){
for(i=0;i<3;i++)V[c[f[x]]][i].del(a[f[x]][i]);
ins(T[f[x]][X],1,m,fe[x],-1);
ins(T[f[x]][y],1,m,fe[x],1);
for(i=0;i<3;i++)V[c[f[x]]][i].add(a[f[x]][i]=ask(T[f[x]][i]));
}
for(i=0;i<3;i++)V[X][i].del(a[x][i]),V[y][i].add(a[x][i]);
for(j=st[x];j<=en[x];j++){
Y=c[q[j][0]];
if(X<Y)V[X][Y].del(q[j][1]);else V[Y][X].del(q[j][1]);
if(y<Y)V[y][Y].add(q[j][1]);else V[Y][y].add(q[j][1]);
}
c[x]=y;
}
}
return 0;
}

  

BZOJ3009 : 集合的更多相关文章

  1. java基础集合经典训练题

    第一题:要求产生10个随机的字符串,每一个字符串互相不重复,每一个字符串中组成的字符(a-zA-Z0-9)也不相同,每个字符串长度为10; 分析:*1.看到这个题目,或许你脑海中会想到很多方法,比如判 ...

  2. .Net多线程编程—并发集合

    并发集合 1 为什么使用并发集合? 原因主要有以下几点: System.Collections和System.Collections.Generic名称空间中所提供的经典列表.集合和数组都不是线程安全 ...

  3. 一起学 Java(三) 集合框架、数据结构、泛型

    一.Java 集合框架 集合框架是一个用来代表和操纵集合的统一架构.所有的集合框架都包含如下内容: 接口:是代表集合的抽象数据类型.接口允许集合独立操纵其代表的细节.在面向对象的语言,接口通常形成一个 ...

  4. 编写高质量代码:改善Java程序的151个建议(第5章:数组和集合___建议75~78)

    建议75:集合中的元素必须做到compareTo和equals同步 实现了Comparable接口的元素就可以排序,compareTo方法是Comparable接口要求必须实现的,它与equals方法 ...

  5. java基础_集合List与Set接口

    List接口继承了Collection的方法  当然也有自己特有的方法向指定位置添加元素   add(索引,添加的元素); 移除指定索引的元素   remove(索引) 修改指定索引的元素   set ...

  6. Java基础Collection集合

    1.Collection是所有集合的父类,在JDK1.5之后又加入了Iterable超级类(可以不用了解) 2.学习集合从Collection开始,所有集合都继承了他的方法 集合结构如图:

  7. 轻量级“集合”迭代器-Generator

    Generator是PHP 5.5加入的新语言特性.但是,它似乎并没有被很多PHP开发者广泛采用.因此,在我们了解PHP 7对Generator的改进之前,我们先通过一个简单却显而易见的例子来了解下G ...

  8. Asp.net MVC 传递数据 从前台到后台,包括单个对象,多个对象,集合

    今天为大家分享下 Asp.net MVC 将数据从前台传递到后台的几种方式. 环境:VS2013,MVC5.0框架 1.基本数据类型 我们常见有传递 int, string, bool, double ...

  9. 这些.NET开源项目你知道吗?.NET平台开源文档与报表处理组件集合(三)

    在前2篇文章这些.NET开源项目你知道吗?让.NET开源来得更加猛烈些吧 和这些.NET开源项目你知道吗?让.NET开源来得更加猛烈些吧!(第二辑)中,大伙热情高涨.再次拿出自己的私货,在.NET平台 ...

随机推荐

  1. python BeautifulSoup

    之前解析LXML,用的是XPath,现在临时被抓取写爬虫,接人家的代码,看到用的是BeautifulSoup,稍微学了下,也挺好用的,简单记录下用法,有机会做下和Xpath的对比测试 初始化 from ...

  2. php-fpm的status可以查看汇总信息和详细信息

    nginx.conf 配置文件 server { listen ; server_name localhost; index index.php index.html; root /home/tiny ...

  3. [整理]win7下VS2010遇到内存不足解决方发

    电脑重装Win7 64bit不久后,一天内VS2010使用久了,就会出现内存不足,实际内存使用情况却不是,显示内存已使用70%.以前没有遇到过,经网上查找,貌似是VS2010对内存计算会在某些情况下计 ...

  4. c++刷题(33/100)笔试题1

    笔试总共2小时,三道题,时间挺充裕的,但是最后只做了一道,原因在于自己很浮躁,不审题,不仔细思考.没过的两道都是稍微改一下代码就能过,但是没过就是没过,要引以为戒 题目1: 小W有一个电子时钟用于显示 ...

  5. 蓝桥杯 问题 1110: 2^k进制数 (排列组合+高精度巧妙处理)

    题目链接 题目描述 设r是个2^k 进制数,并满足以下条件: (1)r至少是个2位的2^k 进制数. (2)作为2^k 进制数,除最后一位外,r的每一位严格小于它右边相邻的那一位. (3)将r转换为2 ...

  6. asp.net(c#)中相对路径(虚拟路径)和物理磁盘路径的转换

    物理路径:磁盘路径,也就是在磁盘上的位置. 虚拟路径:web页面上的路径,是相对于应用程序而言的. /// 将物理路径转换成相对路径           /// </summary>   ...

  7. 关于webpack下热更新?&自动刷新?的小记(非vue-cli)

    写本随笔时:webpack4.6.0 为何标题用?号,因为老衲也不知是否用词正确,大概是这样的说法: webpack4.0引入生产模式和开发模式,在开发时使用 webpack 打包后不压缩,所以只需要 ...

  8. request_irq与request_threaded_irq

    /* * Allocate the IRQ */ #if 0 retval = request_irq(uap->port.irq, pl011_int, 0, "uart-pl011 ...

  9. 一个无锁消息队列引发的血案(五)——RingQueue(中) 休眠的艺术

    目录 (一)起因 (二)混合自旋锁 (三)q3.h 与 RingBuffer (四)RingQueue(上) 自旋锁 (五)RingQueue(中) 休眠的艺术 (六)RingQueue(中) 休眠的 ...

  10. jexus - 分析日志文件

    1.统计IP访问次数 awk '{print $3}' default |sort -n|uniq -c|sort -rn|head