题目链接

题意

给定一个无向图,\(n\)个点,\(m\)条边(\(n,m\leq 1e5\)).

重复如下操作:

选择相异的两点u,v满足从点u出发走三条边恰好能到达点v。在这样的u,v点对之间添一条边(如果已经存在则无需再次添加)。

问最多能添加多少条边。

我的思路

(是写给自己看的读者老爷可以跳过去的部分)

首先, 所有距离为奇数的点对之间都能添边。因为

3-1+3=5
5-1+3=7
5-1+5=9
7-1+3=9
7-1+5=11
7-1+7=13
9-1+3=11
9-1+5=13
...

其次,如果距离为\(2\)的两点之间还有一条边,那么好像能够添成一个完全图……

然后就不会写了(卒

结论

如果该图为二分图,则最多能在所有黑点和白点之间加边;

如果不为二分图,则能加成一个完全图。

证明

官方题解

假设在\(s\)和\(t\)之间存在一条长度为奇数的路径,也就是说对于某个奇数\(k\),存在一个点的序列\(s=v_0,v_1,...,v_k=t\)(\(v_i\)和\(v_{i+1}\)相邻)。

如果\(k=1\),那么\(s\)和\(t\)之间本身就有一条边。如果\(k\geq 3\),通过在\(v_{k-3}\)和\(v_k\)之间加边,我们得到了一条\(s\)和\(t\)之间长度为\(k-2\)的路径。通过重复这个操作,我们可以一直减少\(s\)和\(t\)之间路径的长度最后在\(s\)和\(t\)之间加一条边。

这表示了奇数长路径的重要性,以及二分图性质的重要性。

Case 1. 不是二分图

不是二分图意味着存奇数长度的环,不妨设点\(v\)为环中的一点。因为图是连通图,所以对于任意的点对\((s,t)\),总有一条路径\(s\rightarrow v\rightarrow t\). 如果路径长度为偶数,则可通过加上点\(v\)处的奇环使得路径长度为奇数。

因此,在任意两点间都能找到一条奇数长度的路径,故可以将图加成一个完全图。所以答案为\(N(N-1)/2-M\).

Case 2. 是二分图

这种情况下,可以将点染成黑点和白点,每条边连接着一个白点和一个黑点。考虑任意的黑点和白点,因为图是连通图,所以黑点\(b\)和白点\(w\)之间存在着一条路径,并且显然路经长为奇数。因此,可以在\(b\)和\(w\)之间加边。另一方面,在同种颜色的点之间绝对无法加边。

故答案为\(BW-M\),\(B\)为白点个数,\(W\)为黑点个数。

具体写法

二分图的充要条件:所有回路长度均为偶数

法一:\(dfs\)

模拟染色看是否一个顶点会染到不同的颜色

法二(很新颖):并查集

具体解释见noip 2010 关押罪犯这道题的并查集做法

Code

Ver. 1 dfs

#include <bits/stdc++.h>
#define maxn 100010
using namespace std;
typedef long long LL;
struct Edge {
int to, ne;
Edge(int _to=0, int _ne=0): to(_to), ne(_ne) {}
}edge[maxn*2];
int tot, ne[maxn];
void add(int u, int v) {
edge[tot] = Edge(v, ne[u]);
ne[u] = tot++;
}
int n, m, c[maxn];
void dfs(int u, int col) {
c[u] = col;
for (int i = ne[u]; ~i; i = edge[i].ne) {
int v = edge[i].to;
if (c[v] == -1) dfs(v, !col);
if (c[v] != !col) { printf("%lld\n", 1LL*n*(n-1)/2-m); exit(0); }
}
}
int main() {
scanf("%d%d", &n, &m);
memset(c, -1, sizeof(c));
memset(ne, -1, sizeof(ne));
for (int i = 0; i < m; ++i) {
int u, v;
scanf("%d%d", &u, &v);
add(u, v);
add(v, u);
}
dfs(1, 0);
int b=0, w=0;
for (int i = 1; i <= n; ++i) if (c[i]) ++b; else ++w;
printf("%lld\n", 1LL*b*w-m);
return 0;
}

Ver. 2 并查集

#include <bits/stdc++.h>
#define maxn 200010
using namespace std;
typedef long long LL;
int fa[maxn], sz[maxn];
int find(int x) { return fa[x] == x ? x : fa[x] = find(fa[x]); }
bool same(int u, int v) { return find(u) == find(v); }
void unionn(int x, int y) {
x = find(x), y = find(y);
if (sz[x] > sz[y]) swap(x, y);
fa[x] = y; sz[y] += sz[x];
}
int main() {
int n, m;
scanf("%d%d", &n, &m);
for (int i = 1; i <= (n<<1); ++i) fa[i] = i;
for (int i = 0; i < m; ++i) {
int u, v;
scanf("%d%d", &u, &v);
unionn(u, n+v);
unionn(n+u, v);
}
for (int i = 1; i <= n; ++i) if (same(i, n+i)) { printf("%lld\n", 1LL*n*(n-1)/2-m); return 0; }
int b = 1, w = 0;
for (int i = 2; i <= n; ++i) if (same(1, i)) ++b; else ++w;
printf("%lld\n", 1LL * b * w - m);
return 0;
}

Atcoder CODE FESTIVAL 2017 qual B C - 3 Steps 二分图的更多相关文章

  1. [AtCoder Code Festival 2017 QualB C/At3574] 3 Steps - 二分图染色,结论

    给你一个n个点m条边的无向图,进行以下操作 如果存在两个点u和v,使得从u走三步能恰好到达v,那么在u和v之间连接一条边 重复这个操作直到不能再连接新的边,问最后有多少条边? n, m <= 1 ...

  2. CODE FESTIVAL 2017 qual B C - 3 Steps【二分图】

    CODE FESTIVAL 2017 qual B C - 3 Steps 题意:给定一个n个结点m条边的无向图,若两点间走三步可以到,那么两点间可以直接连一条边,已经有边的不能连,问一共最多能连多少 ...

  3. Atcoder CODE FESTIVAL 2017 qual B D - 101 to 010 dp

    题目链接 题意 对于一个\(01\)串,如果其中存在子串\(101\),则可以将它变成\(010\). 问最多能进行多少次这样的操作. 思路 官方题解 转化 倒过来考虑. 考虑,最终得到的串中的\(' ...

  4. 题解【AtCoder - CODE FESTIVAL 2017 qual B - D - 101 to 010】

    题目:https://atcoder.jp/contests/code-festival-2017-qualb/tasks/code_festival_2017_qualb_d 题意:给一个 01 串 ...

  5. 【题解】Popping Balls AtCoder Code Festival 2017 qual B E 组合计数

    蒟蒻__stdcall终于更新博客辣~ 一下午+一晚上=一道计数题QAQ 为什么计数题都这么玄学啊QAQ Prelude 题目链接:这里是传送门= ̄ω ̄= 下面我将分几个步骤讲一下这个题的做法,大家不 ...

  6. atcoder/CODE FESTIVAL 2017 qual B/B(dfs染色判断是否为二分图)

    题目链接:http://code-festival-2017-qualb.contest.atcoder.jp/tasks/code_festival_2017_qualb_c 题意:给出一个含 n ...

  7. Atcoder CODE FESTIVAL 2017 qual C D - Yet Another Palindrome Partitioning 回文串划分

    题目链接 题意 给定一个字符串(长度\(\leq 2e5\)),将其划分成尽量少的段,使得每段内重新排列后可以成为一个回文串. 题解 分析 每段内重新排列后是一个回文串\(\rightarrow\)该 ...

  8. Atcoder CODE FESTIVAL 2017 qual C C - Inserting 'x' 回文串

    题目链接 题意 给定字符串\(s\),可以在其中任意位置插入字符\(x\). 问能否得到一个回文串,若能,需插入多少个\(x\). 思路 首先统计出现次数为奇数的字符\(cnt\). \(cnt\ge ...

  9. Atcoder CODE FESTIVAL 2017 qual B E - Popping Balls 组合计数

    题目链接 题意 \(A+B\)个球排成一行,左边\(A\)个为红球,右边\(B\)个为蓝球. 最开始可以选择两个数\(s,t\),每次操作可以取左起第\(1\)或\(s\)或\(t\)个球.问有多少种 ...

随机推荐

  1. MFC 菜单编程 -- 总结

    菜单结构 一个菜单栏可以有若干个子菜单,而一个子菜单又可有若干个菜单项.对于菜单栏的子菜单,由左至右从0开始索引.对于特定的子菜单的菜单项,由上至下建立从0开始的索引.访问子菜单和菜单项,均可通过其索 ...

  2. 标准C++(3)重载

    一.函数的重载 c++中同一作用域下能够定义同名的函数(这就叫重载),但必须满足如下要求: 1.函数的参数列表必须不同,可以使参数数量不同,也可以使参数的类型不同,甚至是参数的顺序不同. 2.函数的返 ...

  3. oracle如何保证读一致性 第一弹

    oracle保证读一致性原理   1:undo segment的概念                   当数据库进行修改的时候,需要把保存到以前的old的数据保存到一个地方,然后进行修改,用于保存o ...

  4. Linux系统监视工具

    转自      http://bbs.51cto.com/thread-971896-1.html # 1: top – 查看活动进程的命令TOP工具能够实时显示系统中各个进程的资源占用状况.默认情况 ...

  5. CodeForces 109C 树形DP Lucky Tree

    赶脚官方题解写得挺清楚的说,=_= 注意数据范围用long long,否则会溢出. #include <iostream> #include <cstdio> #include ...

  6. python基础学习笔记——类空间问题以及类之间的关系

    一. 类的空间问题 1.1 何处可以添加对象属性 class A: def __init__(self,name): self.name = name def func(self,sex): self ...

  7. Python选修课第二届Turtle绘图大赛~~画猫猫

    (a)20161401167 夏思敏 20161401179 段梦格 (b)代码执行视频链接 点击查看:Python使用turtle库画猫猫 (c)程序源码 import turtle turtle. ...

  8. MMM的一周计划 准备公告

    (19.6.17——19.6.22) 目前本周还没有过去所以还会更新 第0周 目前博客更新暂定于 [题目难度颜色见洛谷] 1.绿题以上绝对更新 2.黄题可能更新 3.其他估计不会有更新 准备工作 1. ...

  9. mantisbt邮件配置

    PHP.INI里面 [mail function]; For Win32 only.#SMTP = 192.168.0.249SMTP = smtp.163.comsmtp_port = 25 ; F ...

  10. 计算n的阶乘(n!)末尾0的个数

    题目: 给定一个正整数n,请计算n的阶乘n!末尾所含有“0”的个数. 举例: 5!=120,其末尾所含有的“0”的个数为1: 10!= 3628800,其末尾所含有的“0”的个数为2: 20!= 24 ...