博客
关于我
poj 1236(强连通分量分解模板题)
阅读量:793 次
发布时间:2023-03-03

本文共 2940 字,大约阅读时间需要 9 分钟。

问题解析

将问题抽象为有向图模型后,问题可以分为以下两个部分解决:

问题1:最少需要选多少个顶点作为初始点

  • 找到所有强连通分量:每个强连通分量是一个顶点集合,内部任意两点之间都可以相互到达。
  • 构建有向无环图(DAG):将每个强连通分量缩成一个点。
  • 统计DAG中的入度为0的点数:这些点必须作为初始点,才能覆盖整个DAG。
  • 问题2:最少需要添加多少条边

  • 统计DAG中的入度为0点数(k)和出度为0点数(m)
  • 计算最大值:问题2的答案为max(k, m),因为需要连接这些无法到达的点。
  • 代码实现

    #include 
    #include
    #include
    #include
    using namespace std;
    #define pb push_back
    #define mem(a, b) memset(a, b, sizeof(a))
    const int maxn = 100 + 50;
    int n;
    int num;
    struct Edge {
    int to;
    int next;
    } G[maxn * maxn < 1];
    void addEdge(int u, int v) {
    G[num] = {v, head[u]};
    head[u] = num++;
    }
    struct SCC {
    int col[maxn];
    int in[maxn];
    int out[maxn];
    bool vis[maxn];
    vector
    vs;
    void DFS(int u) {
    vis[u] = true;
    for (int i = head[u]; ~i; i = G[i].next) {
    int v = G[i].to;
    if (vis[v] || (i & 1)) {
    continue;
    }
    DFS(v);
    }
    vs.pb(u);
    }
    void RDFS(int u, int k) {
    col[u] = k;
    vis[u] = true;
    for (int i = head[u]; ~i; i = G[i].next) {
    int v = G[i].to;
    if (vis[v] || !(i & 1)) {
    continue;
    }
    RDFS(v, k);
    }
    }
    int scc() {
    mem(vis, false);
    vs.clear();
    for (int i = 1; i <= n; ++i) {
    if (!vis[i]) {
    DFS(i);
    }
    }
    int k = 0;
    mem(vis, false);
    for (int i = vs.size() - 1; i >= 0; --i) {
    if (!vis[vs[i]]) {
    RDFS(vs[i], ++k);
    }
    }
    mem(in, 0);
    mem(out, 0);
    for (int u = 1; u <= n; ++u) {
    for (int i = head[u]; ~i; i = G[i].next) {
    int v = G[i].to;
    if (i & 1) {
    continue;
    }
    if (col[u] != col[v]) {
    out[col[u]]++;
    in[col[v]]++;
    }
    }
    }
    return k;
    }
    };
    void Solve() {
    int k = _scc.scc();
    int zeroIn = 0;
    int zeroOut = 0;
    for (int i = 1; i <= k; ++i) {
    if (_scc.in[i] == 0) {
    zeroIn++;
    }
    if (_scc.out[i] == 0) {
    zeroOut++;
    }
    }
    printf("%d\n", zeroIn);
    if (k == 1) {
    printf("0\n");
    } else {
    printf("%d\n", max(zeroIn, zeroOut));
    }
    }
    void Init() {
    num = 0;
    mem(head, -1);
    }
    int main() {
    while (~scanf("%d", &n)) {
    Init();
    for (int i = 1; i <= n; ++i) {
    int v;
    while (scanf("%d", &v) && v) {
    addEdge(i, v);
    addEdge(v, i);
    }
    }
    Solve();
    }
    return 0;
    }

    代码解释

  • 数据结构:使用Edge结构存储图的边,SCC结构处理强连通分量。
  • 寻找强连通分量:通过DFS和RDFS算法,找到所有强连通分量,并记录每个节点的入度和出度。
  • 计算问题1和2:统计入度为0和出度为0的节点数量,得出问题1和2的答案。
  • 结论

    通过分析强连通分量和DAG的结构,可以有效解决问题1和问题2。代码实现了这一过程,确保了正确性和效率。

    转载地址:http://fuxfk.baihongyu.com/

    你可能感兴趣的文章
    php,nginx重启
    查看>>
    php:$_ENV 和 getenv区别
    查看>>
    PHP:cURL error 60: SSL certificate unable to get local issuer certificate
    查看>>
    PHP:PDOStatement::bindValue参数类型php5和php7问题
    查看>>
    Q媒体播放器.如何播放具有多个音频的视频?
    查看>>
    pickle
    查看>>
    Pickle thread.lock(Pymongo)
    查看>>
    pickle模块
    查看>>
    qYKVEtqdDg
    查看>>
    pid控制
    查看>>
    PID控制介绍-ChatGPT4o作答
    查看>>
    PID控制器数字化
    查看>>
    Qwen-VL项目使用指南
    查看>>
    PIESDKDoNet二次开发配置注意事项
    查看>>
    PIGS POJ 1149 网络流
    查看>>
    PIL Image对图像进行点乘,加上常数(等像素操作)
    查看>>
    PIL Image转Pytorch Tensor
    查看>>
    PIL&QOOT;IOERROR:带有大图像的图像文件被截断(&Q)
    查看>>
    PIL.Image、cv2的img、bytes相互转换
    查看>>
    PIL.Image进行图像融合显示(Image.blend)
    查看>>