【高速公路——Tarjan】

news/2025/2/25 7:02:55

题目

BFS暴力代码 60p

#include <bits/stdc++.h>
using namespace std;

const int N = 1010;
const int M = 10010; 

int g[N][N];
int n, m;
int h[N], e[M], ne[M], idx;

void add(int a, int b)  // 添加一条边a->b
{
    e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ;
}

void bfs(int u)
{
    queue<int> q;
    q.push(u);
    
    while(q.size())
    {
        int t = q.front(); q.pop();
        for(int i = h[t]; ~i; i = ne[i])
        {
            int j = e[i];
            if(g[u][j]) continue;
            g[u][j] = 1;
            q.push(j);
        }
    }
}
int main()
{
    memset(h, -1, sizeof h);
    
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= m; i++)
    {
        int a, b;
        scanf("%d%d", &a, &b);
        add(a, b);
    }
    
    for(int i = 1; i <= n; i++)
        bfs(i);
        
    int ans = 0;
    for(int i = 1; i <= n; i++)
        for(int j = i+1; j <= n; j++)
            if(g[i][j] && g[j][i]) ans++;
    cout << ans;
}

Tarjan代码

#include <bits/stdc++.h>
using namespace std;
using ll = long long;

const int N = 1e4+10;
const int M = 1e5+10;

int h[N], e[M], ne[M], idx; //链式前向星
int dfn[N], low[N], tot; //时间戳 追溯值 分配器
int scc[N], sz[N], cnt; //强连通分量归属 大小 分配器
int stk[N], tt; //栈相关
bool in[N];
int n, m;

void add(int a, int b)  // 添加一条边a->b
{
    e[idx] = b, ne[idx] = h[a], h[a] = idx ++ ;
}

void tarjan(int u)
{
    dfn[u] = ++tot, low[u] = tot;
    stk[++tt] = u; in[u] = 1;
    
    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[j]) //儿子访问过,但仍在栈中,说明成环,拿时间戳
        {
            low[u] = min(low[u], dfn[j]);
        }
    }
    
    if(dfn[u] == low[u]) //时间戳与追溯值一致,x为强连通分量的根节点,开始退栈
    {
        int t; ++cnt;
        do
        {
            t = stk[tt--]; in[t] = 0;
            scc[t] = cnt;
            sz[cnt]++;
        }while(t != u);
    }
}
int main()
{
    memset(h, -1, sizeof h);
    
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= m; i++)
    {
        int a, b;
        scanf("%d%d", &a, &b);
        add(a, b);
    }
    
    for(int i = 1; i <= n; i++)
        if(!dfn[i]) tarjan(i);
    
    ll ans = 0;
    for(int i = 1; i <= cnt; i++)
        ans += 1ll * sz[i] * (sz[i] - 1) / 2;
        
    cout << ans;
}


http://www.niftyadmin.cn/n/5865131.html

相关文章

vue2版本elementUI的table分页实现多选逻辑

1. 需求 我们需要在表格页上实现多选要求&#xff0c;该表格支持分页逻辑。 2. 认识属性 表格属性 参数说明类型可选值默认值data显示的数据array——row-key行数据的 Key&#xff0c;用来优化 Table 的渲染&#xff1b;在使用 reserve-selection 功能与显示树形数据时&…

UE5 Gameplay框架及继承关系详解

文章目录 前言一、核心类及其继承关系二、核心类的职责与协作2.1 Actor & Pawn2.2 Controller2.3 GameMode & GameState2.4 PlayerState2.5 HUD & UI 三、协作流程示例总结 前言 Unreal Engine 5&#xff08;UE5&#xff09;的 Gameplay 框架 是一个高度模块化的系…

OpenCV计算摄影学(1)图像修复(Inpainting)的函数inpaint()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 使用图像中选定区域的邻域来恢复该选定区域。 cv::inpaint 函数是 OpenCV 中用于图像修复&#xff08;Inpainting&#xff09;的一个重要函数。它…

Kafka面试题----Kafka是如何保证顺序消费的

在 Kafka 中&#xff0c;默认情况下消息是按分区进行顺序存储和读取的&#xff0c;但全局顺序消费&#xff08;即所有分区的消息按顺序消费&#xff09;较难实现。下面分别介绍 Kafka 按分区顺序消费以及实现全局顺序消费的相关内容。 按分区顺序消费 Kafka 本身可以保证单个…

机器人“战场”:创新、落地与未来

从1999年的机器管家&#xff0c;2001年的机器人小孩大卫&#xff0c;到2015年拥有自我意识的“查派”&#xff0c;在科幻电影里&#xff0c;人们赋予了对机器人的各种形象和想象。2018年&#xff0c;尽管只是实验室的试验品&#xff0c;但波士顿动力机器狗Spot的视频还是在国内…

python 判断 字符串在字典列表中

在Python中&#xff0c;如果你想判断一个字符串是否存在于一个字典列表中&#xff0c;你可以通过遍历这个列表并检查每个字典是否包含你想要找的字符串键来实现。这里有几种方法可以做到这一点&#xff1a; 方法1&#xff1a;使用any()函数 你可以使用any()函数和字典的get方法…

【MySQL】第九弹---掌握SQL关键操作:更新、删除、插入与聚合分析的秘诀

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】【Linux系统编程】【MySQL】 目录 1 Update 2 Delete 2.1 删除数据 2.2 截断表 3 插入查询结果 4 聚合函数 5 group by子句的使用 1 Update 语法…

【AI+智造】DeepSeek价值重构:当采购与物控遇上数字化转型的化学反应

作者&#xff1a;Odoo技术开发/资深信息化负责人 日期&#xff1a;2025年2月24日 引言&#xff1a;从事企业信息化工作16年&#xff0c;我见证过无数企业从手工台账到ERP系统的跨越。但真正让采购和物控部门脱胎换骨的&#xff0c;是融合了Deepseek AI的Odoo数字化解决方案——…