zoukankan      html  css  js  c++  java
  • 【DFS】【枚举】Gym

    给你一张有向图,问你将任意一条边变成双向后,所能得到的最大强连通分量的大小。

    缩点之后,预处理can(i,j)表示i能到j。

    之后枚举每一条边(u,v),再枚举其他所有点t,如果can(u,t) && can(t,v),则t能和u、v共在一个强连通分量,尝试更新答案。

    #include<cstdio>
    #include<algorithm>
    #include<vector>
    #include<cstring>
    using namespace std;
    bool can[1010][1010];
    struct Edge{
    	int id,v;
    };
    vector<Edge>G[1010],G2[1010];
    vector<int>vs,rG[1010];
    int n,m,K;
    bool used[1010];
    void dfs(int U){
    	used[U]=1;
    	for(int i=0;i<G[U].size();++i){
    		if(!used[G[U][i].v]){
    			dfs(G[U][i].v);
    		}
    	}
    	vs.push_back(U);
    }
    int cmp[1010];
    void rdfs(int U){
    	used[U]=1;
    	cmp[U]=K;
    	for(int i=0;i<rG[U].size();++i){
    		if(!used[rG[U][i]]){
    			rdfs(rG[U][i]);
    		}
    	}
    }
    int a[1010],ans;
    void dfs3(int rt,int U){
    	can[rt][U]=1;
    	for(int i=0;i<G2[U].size();++i){
    		if(!can[rt][G2[U][i].v]){
    			dfs3(rt,G2[U][i].v);
    		}
    	}
    }
    int p,anss[20010];
    int main(){
    	freopen("input.txt","r",stdin);
    	freopen("output.txt","w",stdout);
    	int x,y;
    	scanf("%d%d",&n,&m);
    	for(int i=1;i<=m;++i){
    		scanf("%d%d",&x,&y);
    		G[x].push_back((Edge){i,y});
    		rG[y].push_back(x);
    	}
    	for(int i=1;i<=n;++i){
    		if(!used[i]){
    			dfs(i);
    		}
    	}
    	memset(used,0,sizeof(used));
    	for(int i=vs.size()-1;i>=0;--i){
    		if(!used[vs[i]]){
    			++K;
    			rdfs(vs[i]);
    		}
    	}
    	for(int i=1;i<=n;++i){
    		++a[cmp[i]];
    	}
    	for(int i=1;i<=K;++i){
    		ans=max(ans,a[i]);
    	}
    	for(int i=1;i<=m;++i){
    		anss[i]=i;
    	}
    	p=m;
    	for(int i=1;i<=n;++i){
    		for(int j=0;j<G[i].size();++j){
    			if(cmp[i]!=cmp[G[i][j].v]){
    				G2[cmp[i]].push_back((Edge){G[i][j].id,cmp[G[i][j].v]});
    			}
    		}
    	}
    	for(int i=1;i<=K;++i){
    		dfs3(i,i);
    	}
    	for(int i=1;i<=K;++i){
    		for(int j=0;j<G2[i].size();++j){
    			int tmp=a[i]+a[G2[i][j].v];
    			for(int l=1;l<=K;++l){
    				if(l!=i && l!=G2[i][j].v && can[i][l] && can[l][G2[i][j].v]){
    					tmp+=a[l];
    				}
    			}
    			if(tmp>ans){
    				p=1;
    				anss[p]=G2[i][j].id;
    				ans=tmp;
    			}
    			else if(tmp==ans){
    				anss[++p]=G2[i][j].id;
    			}
    		}
    	}
    	sort(anss+1,anss+p+1);
    	printf("%d
    %d
    ",ans,p);
    	for(int i=1;i<p;++i){
    		printf("%d ",anss[i]);
    	}
    	if(p){
    		printf("%d
    ",anss[p]);
    	}
    	return 0;
    }
  • 相关阅读:
    此网站的安全证书有问题
    WebBrowser控件打开https站点
    C# 调Win32 API SendMessage简单用法及wMsg常量
    sendmessage和postmessage的区别
    C# 使用Win32 API模拟键盘鼠标操作网页
    WebBrowser控件跨域访问页面内容
    获取webbrowser中元素的屏幕坐标
    一行js代码识别Selenium+Webdriver及其应对方案
    用Selenium自动化测试时,让ChromeDriver中不显示“正受到自动测试软件控制”
    Java Selenium Actions模拟鼠标拖动dragAndDrop总结
  • 原文地址:https://www.cnblogs.com/autsky-jadek/p/7171957.html
Copyright © 2011-2022 走看看