Acwing 1175.最大联通子图(tarjan缩点求scc)

简介: 笔记

Acwing 1175.最大连通子图


题意


1.png


输入格式

第一行包含三个整数 N , M , X  。N,M 分别表示图 G 的点数与边数,X 的意义如上文所述;

接下来 M行,每行两个正整数 a,b ,表示一条有向边(a,b) 。

图中的每个点将编号为 1 到 N ,保证输入中同一个(a,b) 不会出现两次。


输出格式

应包含两行。

第一行包含一个整数 K KK ,第二行包含整数 C   m o d   X o


数据范围


2.png


思路


3.png


强连通分量必然是半连通


求解步骤:


先用tarjan算法求出强连通分量并缩点


建图 给边判重(给边判重的原因是 两个半连通子图不相同当且仅当两个子图有某些点不同 而当两个半连通子图只有边不同时,我们认为它们相等)


拓扑图上求最大半连通子图    ⟺    \iff⟺ 求最长无分叉链


求最长无分叉链    ⟺    \iff⟺ 拓扑图上的最长路(scc中点的数量为权重)


代码

// 题目给的 M 为 1e6 但是我们要建两个图 所以 设 M 为 2e6
const int N = 1e5 + 10, M = 2e6 + 10;
int n, m, mod;
int h[N],hs[N],e[M],ne[M],idx;
int low[N],dfn[N];
int scc,timestamp;
stack<int>stk;
bool in_stk[N];
int id[N],siz[N];
int f[N],g[N];
unordered_set<LL>S;
void add(int h[],int a,int b){
  e[idx] = b,ne[idx] = h[a],h[a] = idx++;
}
void tarjan(int u){
  dfn[u] = low[u] = ++timestamp;
  stk.push(u),in_stk[u] = true;
  for(int i = h[u];~i;i = ne[i]){
    int j = e[i];
    if(!dfn[j]){
      tarjan(j);
      low[u] = min(low[u],low[j]);
    }
    else if(in_stk[j]){
      low[u] = min(low[u],dfn[j]);
    }
  }
  if(dfn[u] == low[u]){
    int y = 0;
    ++scc;
    do{
      y = stk.top();
      stk.pop();
      in_stk[y] = false;
      siz[scc]++;
      id[y] = scc;
    }while(y != u);
  }
}
void solve() {
  scanf("%d%d%d",&n,&m,&mod);
  memset(h,-1,sizeof h);
  memset(hs,-1,sizeof hs);
  while(m--){
    int a,b;scanf("%d%d",&a,&b);
    add(h,a,b);
  }
  for(int i = 1;i <= n;++i){
    if(!dfn[i])
      tarjan(i);
  }
  for(int i = 1; i <= n;++i){ // 对scc建图
    for(int j = h[i];~j;j = ne[j]){
      int k = e[j];
      int a = id[i], b = id[k];
      LL hash = a * 1000000ll + b;
      if(a != b && !S.count(hash)){
        add(hs,a,b);
        S.insert(hash);
      }
    }
  }
  for(int i = scc;i;--i){ // scc递减的顺序为 拓扑序
    if(!f[i]){
      f[i] = siz[i];
      g[i] = 1;
    }
    for(int j = hs[i];~j;j = ne[j]){
      int k = e[j];
      if(f[k] < f[i] + siz[k]){
        f[k] = f[i] + siz[k];
        g[k] = g[i];
      }
      else if(f[k] == f[i] + siz[k]){
        g[k] = (g[k] + g[i]) % mod;
      }
    }
  }
  LL sum = 0,maxn = 0;
  for(int i = 1; i <= scc;++i){
    if(f[i] > maxn){
      maxn = f[i];
      sum = g[i];
    }
    else if(f[i] == maxn){
      sum = (sum + g[i]) % mod;
    }
  }
  printf("%lld\n",maxn);
  printf("%lld\n",sum);
}
signed main() {
    //int _; cin >> _;
    //while (_--)
        solve();
    return 0;
}



目录
相关文章
|
8月前
【每日一题Day243】LC1595连通两组点的最小成本 | 状态压缩dp
【每日一题Day243】LC1595连通两组点的最小成本 | 状态压缩dp
64 0
|
算法 Android开发 索引
LeetCode 周赛上分之旅 #44 同余前缀和问题与经典倍增 LCA 算法
学习数据结构与算法的关键在于掌握问题背后的算法思维框架,你的思考越抽象,它能覆盖的问题域就越广,理解难度也更复杂。在这个专栏里,小彭与你分享每场 LeetCode 周赛的解题报告,一起体会上分之旅。
87 0
|
3月前
|
算法 决策智能
基于prim算法求出网络最小生成树实现网络社团划分和规划
该程序使用MATLAB 2022a版实现路线规划,通过排序节点权值并运用Prim算法生成最小生成树完成网络规划。程序基于TSP问题,采用遗传算法与粒子群优化算法进行路径优化。遗传算法通过编码、选择、交叉及变异操作迭代寻优;粒子群优化算法则通过模拟鸟群觅食行为,更新粒子速度和位置以寻找最优解。
|
8月前
|
算法
【优选算法】——Leetcode——611. 有效三角形的个数
【优选算法】——Leetcode——611. 有效三角形的个数
|
流计算
电力系统潮流【牛顿-拉夫逊法】(4节点、5节点、6节点、9节点)(Matlab代码实现)
电力系统潮流【牛顿-拉夫逊法】(4节点、5节点、6节点、9节点)(Matlab代码实现)
455 0
|
8月前
太空飞行计划问题(最大流,经典最大权闭合子图问题)
太空飞行计划问题(最大流,经典最大权闭合子图问题)
48 0
有向强连通分量(SCC)
有向强连通分量(SCC)
98 0
|
存储 算法
407. 接雨水 II : 经典 Dijkstra 运用题
407. 接雨水 II : 经典 Dijkstra 运用题
|
安全 算法 Go
有向图的强联通分量(SCC)Tarjan算法
有向图的强联通分量(SCC)Tarjan算法
211 0
离散数学_十章-图 ( 5 ):连通性 - 上
离散数学_十章-图 ( 5 ):连通性 - 上
143 0