程序设计思维与实践 Week8 作业 C - 班长竞选

博客内容介绍了如何解决一个关于班长竞选的问题,涉及到图论和强连通分量的概念。通过建立正负图,利用Kosaraju算法找出强连通分量,然后寻找出度为0的节点,这些节点可能包含最高票数的候选人。最终通过遍历和DFS算法计算每个节点的票数,找到票数最多的候选人。
摘要由CSDN通过智能技术生成

题目描述

大学班级选班长,N 个同学均可以发表意见 若意见为 A B 则表示 A 认为 B 合适,意见具有传递性,即 A 认为 B 合适,B 认为 C 合适,则 A 也认为 C 合适 勤劳的 TT 收集了M条意见,想要知道最高票数,并给出一份候选人名单,即所有得票最多的同学,你能帮帮他吗?
Input
本题有多组数据。第一行 T 表示数据组数。每组数据开始有两个整数 N 和 M (2 <= n <= 5000, 0 <m <= 30000),接下来有 M 行包含两个整数 A 和 B(A != B) 表示 A 认为 B 合适。
Output
对于每组数据,第一行输出 “Case x: ”,x 表示数据的编号,从1开始,紧跟着是最高的票数。 接下来一行输出得票最多的同学的编号,用空格隔开,不忽略行末空格!
Sample Input

2
4 3
3 2
2 0
2 1

3 3
1 0
2 1
0 2

Sample Output

Case 1: 2
0 1
Case 2: 2
0 1 2

思路

我们按照投票作为边建立一张图,这里采用链式前向星存储分别存储正图和负图(与正图边方向相反),首先需要获得各个强连通分量,这里采用Kosaraju:
• 用第一遍 dfs 确定原图的逆后序序列,即x 点遍历完成的次序,即回溯时间,用 f[x] 表示
• 第二遍 dfs 在反图中按照逆后序序列进行遍历, 每次由起点遍历到的点即构成一个 SCC
获得了强连通分量之后缩点,即将一个连通分量作为一个点处理,令 sum[i] 表示第 i 个 SCC 中点的个数票数最多的人一定在出度为0的强连通分量中,也就是反图中入度为0的强连通分量,遍历反图中的所以边,将端点不在一个强连通分量中的边按照连通分量加入图3,最后就可以对图3中的各点进行dfs获得可以到达的点的数量作为该连通分量的票数,记录最大票数, 遍历所有点,如果该点所在的连通分量的ans==mP,则输出该点,注意题目输出要求,最后一个输出后面没有多余空格,我可以在从第二个输出开始,在输出前输出一个空格来与前面一个输出隔开,就避免了多余的空格输出。

代码

#include <cstdio>
#include <cstring>
#include <algorithm>
#include<iostream>
#include <vector>
#include <queue>

using namespace std;
//2 <= n <= 5000, 0 <m <= 30000
const int maxn = 100000 + 10;
int T,n,m,tot1,tot2,tot3,head2[5005],head3[5005],ans[5005],head1[5005],deg[5005],f[5005],sum[5005],fcnt,scnt,c[5005],ss;
bool vis1[5005];
struct edge
{
   
    int f,to,next;
}e1[30005],e2[30005],e3[30005];

void add1(int s,int d)
{
   
    e1[++tot1].to
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值