P1989 无向图三元环计数
P1989 无向图三元环计数 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)
属于是只有大文学家能写出来,我只能抄在积累本上的那种。
考虑给每个点赋权
对于原图上的一条边
不难发现形成的有向图是 DAG,因为如果
所以原来的三元环在新图中一定是这样一个形态:
我们只要枚举
只需枚举
现在,我们将
这是一个有序数对,比较时优先比较第一维再比较第二维,容易看出,因为第二维的存在,权值有了互异性。
如此以来可以保证:新图中每个点的出度不大于
- 如果
,显然新图上 的出度不会超过 ,所以新出度不大于 ; - 如果
,因为 ,因此 大于 的点不会超过 个,大于 的点则更不会超过 个。根据我们的连边规则,新出度也不会大于 。
枚举
更帅的写法是
/*
* @Author: crab-in-the-northeast
* @Date: 2023-01-04 13:51:33
* @Last Modified by: crab-in-the-northeast
* @Last Modified time: 2023-01-04 13:56:20
*/
#include <bits/stdc++.h>
inline int read() {
int x = 0;
bool f = true;
char ch = getchar();
for (; !isdigit(ch); ch = getchar())
if (ch == '-')
f = false;
for (; isdigit(ch); ch = getchar())
x = (x << 1) + (x << 3) + ch - '0';
return f ? x : (~(x - 1));
}
const int maxn = (int)1e5 + 5;
const int maxm = (int)2e5 + 5;
int d[maxn], us[maxm], vs[maxm];
std :: vector <int> G[maxn];
int t[maxn];
int main() {
int n = read(), m = read();
for (int i = 1; i <= m; ++i) {
int u = read(), v = read();
us[i] = u;
vs[i] = v;
++d[u];
++d[v];
}
for (int i = 1; i <= m; ++i) {
int u = us[i], v = vs[i];
if (d[u] > d[v])
std :: swap(u, v);
else if (d[u] == d[v] && u > v)
std :: swap(u, v);
G[u].push_back(v);
}
int ans = 0;
for (int u = 1; u <= n; ++u) {
for (int v : G[u])
t[v] = u;
for (int v : G[u])
for (int w : G[v])
if (t[w] == u)
++ans;
}
printf("%d\n", ans);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库
· SQL Server 2025 AI相关能力初探
· 为什么 退出登录 或 修改密码 无法使 token 失效