[hdu 6184 Counting Stars(三元环计数)
hdu 6184 Counting Stars(三元环计数)
题意:
给一张n个点m条边的无向图,问有多少个\(A-structure\)
其中\(A-structure\)满足\(V=(A,B,C,D)\) && \(E=(AB,BC,CD,DA,AC)\)
显然\(A-structure\)是由两个有公共边的三元环构成的
\(1 <=n <= 1e5\)
\(1 <= m <= min(2e5,n*(n-1)/2)\)
思路:
三元环计数
做法1、
①统计每个点的度数
②入度\(<=sqrt(m)\)的分为第一类,入度\(>sqrt(m)\)的分为第二类
③对于第一类,暴力每个点,然后暴力这个点的任意两条边,再判断这两条边的另一个端点是否连接
因为\(m\)条边最多每条边遍历一次,然后暴力的点的入度\(<=sqrt(m)\),所以复杂度约为\(O(msqrt(m))\)
④对于第二类,直接暴力任意三个点,判断这三个点是否构成环,因为这一类点的个数不会超过\(sqrt(m)\)个,所以复杂度约为\(O(sqrt(m)^3)=O(msqrt(m))\)
⑤判断两个点是否连接可以用set,map和Hash都行,根据具体情况而行
这种做法建的是双向边,常数很大
更优的做法2、建有向边 复杂度为\(O(msqrt(m))\)
对所有边按照两端点的度数定向,度数小的往度数大的走,度数相同则按编号小到大走,这样定向后
可以保证是个有向无环图。
为什么呢,要想定向存在环,则这个环上的点度数必须相同,由于保证了编号从小到大走
所以是不可能存在环的。
这样定向同时还保证了每个点的出度不超过\(sqrt(m)\),很容易证明,如果存在一个点出度超过了\(sqrt(m)\),则说明存在其他\(sqrt(m)\)个点的度数\(>sqrt(m)\),算起来超过边数\(m\)了。
对于这道题,我们在求三元环的时候,统计一下每条边有多少对点能构成三元环,\(C(cnt,2)\)累计一下即可
做法1、
#include<bits/stdc++.h>
#define LL long long
using namespace std;
void read(int &x){
x = 0;
char c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') x = x * 10 + c - '0',c = getchar();
}
const int N = 1e5 + 10;
set<LL> g;
int deg[N];
vector<int> G[N];
int vis[N],vi[N];
int main(){
int n, m, u, v, Sz;
while(scanf("%d%d",&n,&m) != EOF){
Sz = sqrt(m + 0.5);
g.clear();
for(int i = 1;i <= n;i++){
vis[i] = vi[i] = deg[i] = 0;
G[i].clear();
}
for(int i = 0;i < m;i++){
scanf("%d%d",&u,&v);
g.insert(u + 1LL * v * n);
g.insert(v + 1LL * u * n);
deg[u]++,deg[v]++;
G[u].push_back(v);
G[v].push_back(u);
}
LL ans = 0;
for(int u = 1;u <= n;u++){
vis[u] = 1;
for(auto v:G[u]) vi[v] = u;
for(auto v:G[u]){
int cnt = 0;
if(vis[v]) continue;
if(deg[v] <= Sz){
for(auto vv:G[v]){
if(vi[vv] == u) cnt++;
}
}else{
for(auto vv:G[u]){
if(g.find(1LL * v * n + vv) != g.end()) cnt++;
}
}
ans += 1LL * cnt * (cnt - 1) / 2;
}
}
printf("%lld\n",ans);
}
return 0;
}
做法二、
#include<bits/stdc++.h>
#define LL long long
#define P pair<int,int>
using namespace std;
void read(int &x){
x = 0;
char c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') x = x * 10 + c - '0',c = getchar();
}
const int N = 1e5 + 10;
set<LL> g;
int deg[N];
vector<pair<int,int> > G[N];
int vi[N];
int X[N * 2],Y[N * 2],cnt[N],pos[N];
int main(){
int n, m, u, v, Sz;
while(scanf("%d%d",&n,&m) != EOF){
Sz = sqrt(m);
for(int i = 1;i <= n;i++){
vi[i] = deg[i] = pos[i] = 0;
G[i].clear();
}
g.clear();
int tot = 0;
for(int i = 0;i < m;i++){
scanf("%d%d",&X[i],&Y[i]);
u = X[i],v = Y[i];
deg[u]++,deg[v]++;
}
for(int i = 0;i < m;i++){
cnt[i] = 0;
if(deg[X[i]] < deg[Y[i]]) G[X[i]].push_back(make_pair(Y[i],i));
else if(deg[Y[i]] < deg[X[i]]) G[Y[i]].push_back(P(X[i],i));
else{
if(X[i] < Y[i]) G[X[i]].push_back(P(Y[i],i));
else G[Y[i]].push_back(P(X[i],i));
}
}
LL ans = 0;
for(int i = 0;i < m;i++){
u = X[i],v = Y[i];
for(auto vp:G[u]) pos[vp.first] = vp.second,vi[vp.first] = i + 1;
for(auto vp:G[v]){
int vv = vp.first;
if(vi[vv] == i + 1){
cnt[i]++;
cnt[pos[vv]]++;
cnt[vp.second]++;
}
}
}
for(int i = 0;i < m;i++) ans += 1LL * cnt[i] * (cnt[i] - 1) / 2;
printf("%lld\n",ans);
}
return 0;
}
[hdu 6184 Counting Stars(三元环计数)的更多相关文章
- HDU 6184 Counting Stars
Problem Description Little A is an astronomy lover, and he has found that the sky was so beautiful!S ...
- HDU 6184 Counting Stars 经典三元环计数
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6184 题意: n个点m条边的无向图,问有多少个A-structure 其中A-structure满足V ...
- 【刷题】HDU 6184 Counting Stars
Problem Description Little A is an astronomy lover, and he has found that the sky was so beautiful! ...
- Codechef SUMCUBE Sum of Cubes 组合、三元环计数
传送门 好久没有做过图论题了-- 考虑\(k\)次方的组合意义,实际上,要求的所有方案中导出子图边数的\(k\)次方,等价于有顺序地选出其中\(k\)条边,计算它们在哪一些图中出现过,将所有方案计算出 ...
- 【BZOJ5332】[SDOI2018]旧试题(数论,三元环计数)
[BZOJ5332][SDOI2018]旧试题(数论,三元环计数) 题面 BZOJ 洛谷 题解 如果只有一个\(\sum\),那么我们可以枚举每个答案的出现次数. 首先约数个数这个东西很不爽,就搞一搞 ...
- loj#6076「2017 山东一轮集训 Day6」三元组 莫比乌斯反演 + 三元环计数
题目大意: 给定\(a, b, c\),求\(\sum \limits_{i = 1}^a \sum \limits_{j = 1}^b \sum \limits_{k = 1}^c [(i, j) ...
- BZOJ.5407.girls/CF985G. Team Players(三元环计数+容斥)
题面 传送门(bzoj) 传送门(CF) \(llx\)身边妹子成群,这天他需要从\(n\)个妹子中挑出\(3\)个出去浪,但是妹子之间会有冲突,表现为\(i,j\)之间连有一条边\((i,j)\), ...
- LOJ2565 SDOI2018 旧试题 莫比乌斯反演、三元环计数
传送门 这道题的思路似乎可以给很多同时枚举三个量的反演题目提供一个很好的启发-- 首先有结论:\(d(ijk) = \sum\limits_{x|i}\sum\limits_{y|j}\sum\lim ...
- hdu6184 Counting Stars 【三元环计数】
题目链接 hdu6184 题解 题意是让我们找出所有的这样的图形: 我们只需要求出每条边分别在多少个三元环中,记为\(x\),再然后以该点为中心的图形数就是\({x \choose 2}\) 所以我们 ...
随机推荐
- 单表60亿记录等大数据场景的MySQL优化和运维之道 | 高可用架构
015-08-09 杨尚刚 高可用架构 此文是根据杨尚刚在[QCON高可用架构群]中,针对MySQL在单表海量记录等场景下,业界广泛关注的MySQL问题的经验分享整理而成,转发请注明出处. 杨尚刚,美 ...
- VC++读写*.ini配置文件
ini文件(即Initialization file),这种类型的文件中通常存放的是一个程序的初始化信息.ini文件由若干个节(Section)组成,每个Section由若干键(Key)组成,每个Ke ...
- 阿里云linux服务器登录失败,Connection closed
ssh_exchange_identification: read: Connection reset by peer报错如下: [root@izbp17x1~]# ssh admin@139.196 ...
- #warning Incomplete method implementation怎么修改?
#warning Incomplete method implementation怎么修改? 各位朋友,我在做一个表格视图的例子,在tableview方法里总有几个warning:#war ...
- 设计一个方法injectBeforeAsyncSend,能够实现如下功能:在发起异步请求之前打印出请求的类型、URL、method、body、timestamp 等信息。
异步请求逻辑注入 工作中我们需要对异步请求的请求信息打印日志,但是又不能耦合在业务代码中打印.请设计一个方法injectBeforeAsyncSend,能够实现如下功能:在发起异步请求之前打印出请求的 ...
- Unity 游戏框架搭建 (十七) 静态扩展GameObject实现链式编程
本篇本来是作为原来 优雅的QChain的第一篇的内容,但是QChain流产了,所以收录到了游戏框架搭建系列.本篇介绍如何实现GameObject的链式编程. 链式编程的实现技术之一是C#的静态扩展.静 ...
- [转]收集Oracle UNDO诊断信息脚本
使用该脚本可收集与undo相关的信息,在undo表空间出问题时可使用该脚本来诊断. 使用方法: 1.将脚本拷贝到服务器,创建文件保存,文件名可随意取,例如:diag.out 2.以sys用户登录数据库 ...
- C# Console类的方法使用总结
Console类表示控制台应用程序的标准输入流.输出流和错误流. 此类不能被继承,而在Java中,类似的功能则由System.in和System.out来实现了. 一 输出到控制台 输出到控制台就是把 ...
- Layered Architecture 分层架构
分层的价值在于每一层都只代表程序中的某一特定方面.这种限制使每个方面的设计都更具有内聚性,更容易解释. 大多数成功的架构使用的都是包括下面这四个概念层的某个版本
- ES5 实现 ES6 的 class以及extends
ts中是这样的 class Greeter { greeting:string; constructor(message:string){ this.greeting = message; } gre ...