zoukankan      html  css  js  c++  java
  • 【luogu2272】 [ZJOI2007]最大半连通子图 [tarjan 缩点][拓扑排序]

    P2272 [ZJOI2007]最大半连通子图

    首先缩点 缩完点后存在大量重边 排一遍序去重

    然后重新建一个新图 再从入度为0的点一个一个搜 统计并更新答案

    感觉过不了多久我再看就看不懂了 一大坨变量

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 #define ll long long
     4 const int N=100000+5,M=1000000+5,inf=0x3f3f3f3f;
     5 int n,m,mod,in[N],out[N],cntp[N],dis[N],ans1=0,ans2=0;
     6 int idx=0,Bcnt=0,dfn[N],low[N],bl[N],size[N];
     7 bool inst[N];
     8 stack<int>s;
     9 template<class t>void rd(t &x){
    10     x=0;int w=0;char ch=0;
    11     while(!isdigit(ch)) w|=ch=='-',ch=getchar();
    12     while(isdigit(ch)) x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
    13     x=w?-x:x;
    14 }
    15 
    16 struct node{int u,v;}nod[M];
    17 int head[N],tot=0,cnte=0;
    18 struct edge{int u,v,nxt;}e[M];
    19 void add(int u,int v){
    20     e[++tot]=(edge){u,v,head[u]};head[u]=tot;
    21 }
    22 void tarjan(int u){
    23     dfn[u]=low[u]=++idx;
    24     s.push(u),inst[u]=1;
    25     for(int i=head[u],v;i;i=e[i].nxt){
    26         v=e[i].v;
    27         if(!dfn[v]) tarjan(v),low[u]=min(low[u],low[v]);
    28         else if(inst[v]&&dfn[v]<low[u]) low[u]=dfn[v];
    29     }
    30     if(low[u]==dfn[u]){
    31         ++Bcnt;
    32         int v;
    33         do{
    34             v=s.top(),s.pop();
    35             bl[v]=Bcnt,++size[Bcnt],inst[v]=0;
    36         }while(u!=v);
    37     }
    38 }
    39 
    40 bool cmp(node a,node b){return a.u==b.u?a.v<b.v:a.u<b.u;}
    41 int head2[N],tot2;
    42 struct edge2{int v,nxt;}e2[M];
    43 void add2(int u,int v){
    44     e2[++tot2]=(edge2){v,head2[u]};head2[u]=tot2;
    45 }
    46 bool vis[N];
    47 void dfs(int u){
    48     vis[u]=1;
    49     if(!out[u]){
    50         dis[u]=size[u],cntp[u]=1;
    51         ans1=max(ans1,dis[u]);
    52         return;
    53     }
    54     for(int i=head2[u],v;i;i=e2[i].nxt){
    55         v=e2[i].v;
    56         if(!vis[v]) dfs(v);
    57         if(dis[u]<dis[v]+size[u]) dis[u]=dis[v]+size[u],cntp[u]=cntp[v]%mod;
    58         else if(dis[u]==dis[v]+size[u]) cntp[u]=(cntp[u]+cntp[v])%mod;
    59         ans1=max(ans1,dis[u]);
    60     }
    61 }
    62 
    63 int main(){
    64     //freopen("in.txt","r",stdin);
    65     rd(n),rd(m),rd(mod);
    66     for(int i=1,u,v;i<=m;++i) rd(u),rd(v),add(u,v);
    67     memset(dfn,0,sizeof(dfn));
    68     for(int i=1;i<=n;++i)
    69     if(!dfn[i]) tarjan(i);
    70     for(int i=1;i<=tot;++i)
    71     if(bl[e[i].u]!=bl[e[i].v]) nod[++cnte].u=bl[e[i].u],nod[cnte].v=bl[e[i].v];
    72     sort(nod+1,nod+1+cnte,cmp);
    73     for(int i=1;i<=cnte;++i)
    74     if(nod[i].u!=nod[i-1].u||nod[i].v!=nod[i-1].v)
    75     add2(nod[i].u,nod[i].v),++in[nod[i].v],++out[nod[i].u];
    76     for(int i=1;i<=Bcnt;++i)
    77     if(!in[i]&&!vis[i]) dfs(i);
    78     for(int i=1;i<=Bcnt;++i)
    79     if(dis[i]==ans1) ans2=(ans2+cntp[i])%mod;
    80     printf("%d
    %d",ans1,ans2);
    81     return 0;
    82 }
  • 相关阅读:
    常用基础命令
    Vim
    Linux目录结构
    稀疏数组
    数据结构概述
    天天用的命令
    Mysql和redis的安装
    回文排列
    URL化
    在word中做复选框打对勾钩
  • 原文地址:https://www.cnblogs.com/lxyyyy/p/11159909.html
Copyright © 2011-2022 走看看