CF859E 题解
分析
我们不妨把这些座位看作是一张图中的节点,把每个人的诉求作为一条边(由【原座位】指向【想去的座位】)
比如,对于样例#1,我们就可以得到这样一张图:

显然,我们有可能会得到多个连通图(比如上面这张)
因为每个座位上的人都只有一个想去的座位,所以每个点的出度就是 \(0\)(可能只有其他的点指向它) 或 \(1\),整每个连通图的边数就是 \(n\) (点的个数)或者 \(n - 1\) 个
当边数等于 \(n\) 时,显然这是一个环或者一棵基环树(定义见这里)
我们以这组数据举例:
7
1 2
2 4
3 2
4 1
5 3
6 3
7 6
手画一下复制到Graph Editor,可以得到这张图:

试着移动每一个点,容易发现:移动r任意环外点(比如这里的 \(3\)、\(5\)、\(6\)、\(7\)),环上的点肯定就没位置了。
证明(废话) :点 \(2\) 唯一的一条出边已经用在环上了,所以 \(2\) 和任意环外点之间的边一定是指向 \(2\) 的,以此类推,我们可以发现:\(3\)、\(5\)、\(6\)、\(7\) 对应的边都是指向 \(2\) 的。于是,只要移动必定会移到 \(2\) 的位置
于是我们就可以得到结论:如果这是一棵基环树或者一个环,那么它的答案为 \(2\)(环内的点移/不移,两种情况)
当边数等于 \(n\) 时,显然这是一棵树
我们拿它????举例:

每个节点的出度都为 \(1\) 或 \(0\)(只可能有一个点出度为 \(0\),这个座位一开始就空着),所以我们可以将唯一的那个出度为 \(0\) 的点(如这张图中的点 \(0\))作为树根。
显然,我们移动任意一个点,都会导致一种确定且不重复的情况(它到根节点的链上的每一个点都往上移),所以其答案为 \(n\),即点数
于是,最终的答案就是将每个连通图的答案乘起来。
HowTo
我们现在需要一种方法,可以判环、可以记录每个连通图的大小。
显然,并查集很合适。
每输入一个人的要求,我们就把这两个点所在的集合(连通图)合并(merge函数),同时记录集合大小(sz数组)。如果合并前这两个点已经在同一个集合中,那么,这一定是一个环(两点之间有不止一条路径),做一下标记。
最后,统计答案即可。
你以为结束了?
还有一个坑。。。
出现自环时,不能向上面那样考虑,因为这一个点移动后还在原来的位置,所以答案是 \(1\)(最终ans不变)
细节请看代码
#include <bits/stdc++.h>
using namespace std;
#define int long long // 好习惯:随时开long long
#define MAXN 200005
#define MOD 1000000007 // 记得取模
int n;
int fa[MAXN]; // 并查集 - 父节点
int sz[MAXN]; // 并查集 - 集合大小
bool ish[MAXN]; // is环(记录此集合是否有环)
bool iszh[MAXN]; // is自环
void init(int x) { // 并查集 - 初始化
for (int i = 1; i <= x; i++) {
fa[i] = i;
sz[i] = 1; // 初始大小都是1
}
}
int find(int x) { // 并查集 - 查找
if (fa[x] == x) {
return x;
} else {
return fa[x] = find(fa[x]);
}
}
void merge(int x, int y) { // 并查集 - 合并
int fx = find(x);
int fy = find(y);
if (fx == fy) { // 已经在同一个集合
return;
}
iszh[fy] |= iszh[fx]; // 合并自环信息
sz[fy] += sz[fx]; // 合并大小
fa[fx] = fy;
}
signed main() {
cin >> n;
init(n * 2); // 初始化
for (int i = 1; i <= n; i++) {
int a, b;
cin >> a >> b;
if (a == b) {
iszh[find(a)] = 1; // 标记当前集合含有自环
}
if (find(a) == find(b)) { // 已经在同一个集合了
ish[find(a)] = 1; // 标记当前集合含有环
}
merge(a, b); // 连接这两个座位
}
int ans = 1;
for (int i = 1; i <= n * 2; i++) {
if (find(i) == i) { // i是一个集合的代表
if (ish[i]) { // 含有环
if (iszh[i]) { // 自环,答案不变
continue;
}
ans *= 2;
ans %= MOD;
} else { // 不含有环
ans *= sz[i];
ans %= MOD;
}
}
}
cout << ans << endl;
}
AC
CF859E 题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- 浅析kubernetes中client-go Informer
之前了解了client-go中的架构设计,也就是 tools/cache 下面的一些概念,那么下面将对informer进行分析 Controller 在client-go informer架构中存在一 ...
- 「JOISC 2020 Day1」汉堡肉
我终于学会打开机房的LOJ了! description LOJ3272 有\(n(n<=2*10^5)\)个矩形,让你找\(k(k<=4)\)个点可以覆盖所有矩形(点可重复),输出一种方案 ...
- 《Unix 网络编程》11:名字和地址转换
名字和地址转换 系列文章导航:<Unix 网络编程>笔记 域名系统 简介 域名系统主要用于主机名字和 IP 地址之间的映射.主机名可以是: 简单名字,如:centos01 全限定域名(FQ ...
- LowDB采坑记录(主要为lowdb3.0的Cannot find module和lowdb1.0 node不断自启动的问题)
bug1: Error [ERR_ MODULE_ NOT_ FOUND]: Cannot find module 在使用最新版LowDB(3.0)时,发现typescript中直接引用模块名(如以下 ...
- java中关于@override注解的使用
@Override是伪代码,表示重写,作用有:1.可以当注释用,方便阅读:2.编译器可以给你验证@Override下面的方法名是否是你父类中所有的,如果没有则报错.例如:如果想重写父类的方法,比如to ...
- 我是一个Dubbo数据包...
hello,大家好呀,我是小楼! 今天给大家带来一篇关于Dubbo IO交互的文章,本文是一位同事写的文章,用有趣的文字把枯燥的知识点写出来,通俗易懂,非常有意思,所以迫不及待找作者授权然后分享给大家 ...
- python爬虫之JS逆向
Python爬虫之JS逆向案例 由于在爬取数据时,遇到请求头限制属性为动态生成,现将解决方式整理如下: JS逆向有两种思路: 一种是整理出js文件在Python中直接使用execjs调用js文件(可见 ...
- 开发工具-在线计算MD5
更新记录: 2022年6月8日 更新标题. 2022年6月1日 开始. 都记在这以后就不用到处找了. 在线计算MD5 https://www.sojson.com/md5/ http://www.ip ...
- 聊聊 C# 和 C++ 中的 泛型模板 底层玩法
最近在看 C++ 的方法和类模板,我就在想 C# 中也是有这个概念的,不过叫法不一样,人家叫模板,我们叫泛型,哈哈,有点意思,这一篇我们来聊聊它们底层是怎么玩的? 一:C++ 中的模板玩法 毕竟 C+ ...
- NC20241 [SCOI2005]扫雷MINE
NC20241 [SCOI2005]扫雷MINE 题目 题目描述 相信大家都玩过扫雷的游戏.那是在一个 \(n \times m\) 的矩阵里面有一些雷,要你根据一些信息找出雷来. 万圣节到了 ,&q ...