图论三
0、目录
无向图割顶和桥、无向图双连通分量、有向图强连通分量、TwoSAT、并查集、拓扑排序、黑白染色、欧拉图、表达式树
(参考自白皮)
1、无向图割顶和桥
int pre[maxn],low[maxn],iscut[maxn],dfs_clock;
//int isbrige[maxm];
int dfs(int u,int fa){
int lowu=pre[u]=++dfs_clock;
int child=0;
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
if(!pre[v]){
child++;
int lowv=dfs(v,u);
lowu=min(lowu,lowv);
if(lowv>=pre[u]){
iscut[u]=true;
}
// if(lowv>pre[u]){
// isbridge[e]=true;
// }
}else if(pre[v]<pre[u]&&v!=fa){
lowu=min(lowu,pre[v]);
}
}
if(fa<0&&child==1) iscut[u]=0;
low[u]=lowu;
return lowu;
}
2、无向图双连通分量
2.1、点双联通
int pre[maxn],iscut[maxn],bccno[maxn],dfs_clock,bcc_cnt;
vector<int> G[maxn],bcc[maxn];
stack<Edge> S;
int dfs(int u,int fa){
int lowu=pre[u]=++dfs_clock;
int child=0;
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
Edge& e=Edge(u,v);
if(!pre[v]){
S.push(e);
child++;
int lowv=dfs(v,u);
lowu=min(lowu,lowv);
if(lowv>=pre[u]){
iscut[u]=true;
bcc_cnt++; bcc[bcc_cnt].clear();
for(;;){
Edge& x=S.top(); S.pop();
if(bccno[x.u]!=bcc_cnt){
bcc[bcc_cnt].push_back(x.u);
bccno[x.u]=bcc_cnt;
}
if(bccno[x.v]!=bcc_cnt){
bcc[bcc_cnt].push_back(x.v);
bccno[x.v]=bcc_cnt;
}
if(x.u==u&&x.v==v) break;
}
}
}else if(pre[v]<pre[u]&&v!=fa){
S..push(e);
lowu=min(lowu,pre[v]);
}
}
if(fa<0&&child==1) iscut[u]=0;
return lowu;
}
void find_bcc(int n){
memset(pre,0,sizeof(pre));
memset(iscut,0,sizeof(iscut));
memset(bccno,0,sizeof(bccno));
dfs_clock=bcc_cnt=0;
for(int i=0;i<n;i++){
if(!pre[i]) dfs(i,-1);
}
}
2.2、边双连通
2.2.1、把桥都删了,剩下的连通分量就是边双连通
2.2.2、用有向图强连通分量来处理
3、有向图强连通分量
3.1、两次dfs
vector<int> G[maxn],G2[maxn];
vector<int> S;
int vis[maxn],sccno[maxn],scc_cnt;
void dfs1(int u){
if(vis[u]) return ;
vis[u]=1;
for(int i=0;i<G[u].size();i++) dfs1(G[u][i]);
S.push_back(u);
}
void dfs2(int u){
if(sccno[u]) return;
sccno[u]=scc_cnt;
for(int i=0;i<G2[u].size();i++) dfs2(G2[u][i]);
}
void find_scc(int n){
scc_cnt=0;
S.clear();
memset(sccno,0,sizeof(sccno));
memset(vis,0,sizeof(vis));
for(int i=0;i<n;i++) dfs1(i);
for(int i=n-1;i>=0;i--)
if(!sccno[S[i]]){ scc_cnt++; dfs2(S[i]); }
}
3.2、tarjan
vector<int> G[maxn];
int pre[maxn],lowlink[maxn],sccno[maxn],dfs_clock,scc_cnt;
stack<int> S;
void dfs(int u){
pre[u]=lowlink[u]=++dfs_clock;
S.push(u);
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
if(!pre[v]){
dfs(v);
lowlink[u]=min(lowlink[u],lowlink[v]);
}else if(!sccno[v]){
lowlink[u]=min(lowlink[u],pre[v]);
}
}
if(lowlink[u]==pre[u]){
scc_cnt++;
for(;;){
int x=S.top(); S.pop();
sccno[x]=scc_cnt;
if(x==u) break;
}
}
}
void find_scc(int n){
dfs_clock=scc_cnt=0;
memset(sccno,0,sizeof(sccno));
memset(pre,0,sizeof(pre));
for(int i=0;i<n;i++) if(!pre[i]) dfs(i);
}
4、TowSAT
struct TwoSAT{
int n;
vector<int> G[maxn*2];
bool mark[maxn*2];
int S[maxn*2],c;
bool dfs(int x){
if(mark[x^1]) return false;
if(mark[x]) return true;
mark[x]=true;
S[c++]=x;
for(int i=0;i<G[x].size();i++)
if(!dfs(G[x][i])) return false;
return true;
}
void init(int n){
this->n=n;
for(int i=0;i<n*2;i++) G[i].clear();
memset(mark,0,sizeof(mark));
}
void add_clause(int x,int xval,int y,int yval){
x=x*2+xval;
y=y*2+yval;
G[x^1].push_back(y);
G[y^1].push_back(x);
}
bool solve(){
for(int i=0;i<n*2;i+=2){
if(!mark[i]&&!mark[i+1]){
c=0;
if(!dfs(i)){
while(c>0) mark[S[--c]]=false;
if(!dfs(i+1)) return false;
}
}
}
return true;
}
}
5、并查集
int fa[maxn];
int find(int x){
return fa[x]=fa[x]==x?x:find(fa[x]);
}
int Union(int u,int v){
int pu=find(u);
int pv=find(v);
if(pu!=pv){
fa[pv]=pu;
}
}
6、拓扑排序
queue<int> Q;
for(int i=0;i<n;i++){
if(!ind[i]) Q.push(i);
}
while(!Q.empty()){
int u=Q.front(); Q.pop();
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
ind[v]--;
if(!ind[v]){
Q.push(v);
}
}
}
7、黑白染色
int color[maxn];
bool dfs(int u){
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
if(color[v]==color[u]) return false;
if(!color[v]){
color[v]=3-color[u];
if(!dfs(v)) return false;
}
}
}
8、欧拉路
8.1、无向图欧拉回路构造
int vis[maxm];
vector<int> path;
void euler(int u){
for(int i=0;i<G[u].size();i++){
int id=G[u][i];
Edge& e=egs[id];
if(!vis[id]){
vis[id]=vis[id^1]=1;
euler(e.v);
path.push_back(e.v);
}
}
}