题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3569
Description
Input
Output
Sample Input
2 1
3 2
4 2
5 1
5 3
4 1
4 3
5 2
3 1
5 4
5
1 1
3 7 0 3
4 0 7 4 6
2 2 7
4 5 0 2 13
Sample Output
Connected
Connected
Connected
Disconnected
HINT
N≤100000,M≤500000,Q≤50000,1≤K≤15,数据保证没有重边与自环
Tip:请学会使用搜索引擎
题意概述:
给出一张图,每次假设删除图上K条边,询问图是否连通。
分析:
这操作还是很厉害的......
对于图的问题可以借助和图有关的树来分析。
借助DFS树,我们发现当我们删除一些边的时候,只有我们把某条树边以及所有跨越它的非树边删除掉之后这个图就不连通了。
那么我先现在需要判断给出的边中有没有这样的一些边出现。如果有,那么图就不连通,否则就依旧连通。
怎么判断呢?想到异或,当一些线性相关的数字一起出现的时候,它们的其中一些异或和为0。对于每一条非树边,将其随机一个权值,然后对其跨越的所有边异或上它的权值。每一次询问的时候取出来求线性基,如果线性基插入过程中有数字变成了0,说明给出的数字不是线性不相关。
因为K<=15,所以说可以直接用rand()函数(实在不放心可以手动随机二进制位)。
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<cstdlib> 5 #include<algorithm> 6 #include<cmath> 7 #include<queue> 8 #include<set> 9 #include<map> 10 #include<vector> 11 #include<cctype> 12 using namespace std; 13 const int maxn=100005; 14 const int maxm=500005; 15 16 int N,M,Q; 17 struct edge{ int to,next; }E[maxm<<1]; 18 int first[maxn],np,fl[maxn],delt[maxn],dfn[maxn],dfs_clock,w[maxm],vis[maxn]; 19 struct Linear_Base{ 20 static const int up=16; 21 int b[up]; 22 void init(){ memset(b,0,sizeof(b)); } 23 bool ins(int x){ 24 for(int i=up-1;i>=0;i--) if((1<<i)&x){ 25 if(!b[i]) { b[i]=x; break; } 26 else x^=b[i]; 27 } 28 return x!=0; 29 } 30 }lb; 31 32 void add_edge(int u,int v) 33 { 34 E[++np]=(edge){v,first[u]}; 35 first[u]=np; 36 } 37 void data_in() 38 { 39 scanf("%d%d",&N,&M); 40 int x,y; 41 for(int i=1;i<=M;i++){ 42 scanf("%d%d",&x,&y); 43 add_edge(x,y); add_edge(y,x); 44 } 45 scanf("%d",&Q); 46 } 47 void DFS(int i,int fp) 48 { 49 dfn[i]=++dfs_clock,fl[i]=fp; 50 for(int p=first[i];p;p=E[p].next){ 51 if(fp==(p-1^1)+1) continue; 52 int j=E[p].to; 53 if(dfn[j]){ 54 if(dfn[j]<dfn[i]){ 55 w[p+1>>1]=rand()+1; 56 delt[i]^=w[p+1>>1],delt[j]^=w[p+1>>1]; 57 } 58 continue; 59 } 60 DFS(j,p); 61 w[fp+1>>1]^=w[p+1>>1]; 62 } 63 w[fp+1>>1]^=delt[i]; 64 } 65 void work() 66 { 67 srand(20010207); 68 DFS(1,0); 69 int k,x,cnt=0; 70 for(int i=1;i<=Q;i++){ 71 scanf("%d",&k); 72 bool ok=1; lb.init(); 73 for(int j=1;j<=k;j++){ 74 scanf("%d",&x); 75 if(ok&&!lb.ins(w[x^cnt])) ok=0; 76 } 77 if(ok) puts("Connected"),cnt++; 78 else puts("Disconnected"); 79 } 80 } 81 int main() 82 { 83 data_in(); 84 work(); 85 return 0; 86 }