LINK


题目大意

有一群人,其中有一些人之间有矛盾,现在要求选出一些人形成一个环,这个环要满足如下条件:

1.人数大于1

2.总人数是奇数

3.有矛盾的人不能相邻

问有多少人不能和任何人形成任何的环

思路

发现如果在原图上直接判断非常的不方便

考虑在补图上挖掘性质

补图:

补图和原图没有任何一条重边

不图和原图的所有边并集是一个完全图

即删去所有存在的边,把不存在的边加上

发现可以选出来的环在补图上一定是一个奇环

那么就可以考虑怎么找到补图中的所有奇环

  • 性质1:包含一个奇环的点双连通分量中的每一个点一定属于某一个奇环

这个性质还挺神奇的

因为这个点双联通分量中的每两个点之间至少有两条点不相交路径

所以假设一个偶环包含了一个奇环中的一部分\(p_u...p_v\),

因为p是奇环所以\(p_u\)到\(p_v\)中一定有一条长度是奇数的路径和一条长度是偶数的路径

所以结论成立咯

所以就可以发现只需要对每个强连通分量判断包不包含奇环就可以了

  • 性质2:包含奇环的充分必要条件是二分图染色冲突

还挺好证明的吧

如果二分图染色不出现冲突,就是个二分图了

二分图中是没有奇环的

所以直接点双联通判一下有没有奇环就可以了


毒瘤poj上有多组数据

注意清零的问题

特别是二分图染色的数组


//Author: dream_maker
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<stack>
using namespace std;
//----------------------------------------------
//typename
typedef long long ll;
//convenient for
#define fu(a, b, c) for (int a = b; a <= c; ++a)
#define fd(a, b, c) for (int a = b; a >= c; --a)
#define fv(a, b) for (int a = 0; a < (signed)b.size(); ++a)
//inf of different typename
const int INF_of_int = 1e9;
const ll INF_of_ll = 1e18;
//fast read and write
template <typename T>
void Read(T &x) {
bool w = 1;x = 0;
char c = getchar();
while (!isdigit(c) && c != '-') c = getchar();
if (c == '-') w = 0, c = getchar();
while (isdigit(c)) {
x = (x<<1) + (x<<3) + c -'0';
c = getchar();
}
if (!w) x = -x;
}
template <typename T>
void Write(T x) {
if (x < 0) {
putchar('-');
x = -x;
}
if (x > 9) Write(x / 10);
putchar(x % 10 + '0');
}
//----------------------------------------------
const int N = 1e3 + 10;
const int M = 1e6 + 10;
struct Edge{
int u, v, nxt;
} E[M << 1];
int g[N][N], n, m, in[N];
int head[N], tot, vis[N], col[N];
void add(int u, int v) {
++tot;
E[tot].u = u;
E[tot].v = v;
E[tot].nxt = head[u];
head[u] = tot;
}
void clean() {
memset(head, 0, sizeof(head));
tot = 0;
}
bool dfs(int u, int now) {
col[u] = now;
for (int i = head[u]; i; i = E[i].nxt) {
int v = E[i].v;
if (!in[v]) continue;
if (col[v] && col[v] == col[u]) return 1;
else if (!col[v] && dfs(v, 3 - now)) return 1;
}
return 0;
}
namespace Tarjan {
int dfn[N], low[N], bel[N];
int ind = 0, cnt_bcc = 0;
stack<Edge> st;
vector<int> bcc[N];
void init() {
memset(dfn, 0, sizeof(dfn));
memset(low, 0, sizeof(low));
memset(bel, 0, sizeof(bel));
ind = 0;
}
void tarjan(int u, int fa) {
dfn[u] = low[u] = ++ind;
for (int i = head[u]; i; i = E[i].nxt) {
int v = E[i].v;
if (v == fa) continue;
if (!dfn[v]) {
st.push(E[i]);
tarjan(v, u);
low[u] = min(low[u], low[v]);
if (low[v] >= dfn[u]) {
++cnt_bcc;
bcc[cnt_bcc].clear();
Edge now;
do {
now = st.top(); st.pop();
if (bel[now.u] != cnt_bcc) {
bel[now.u] = cnt_bcc;
bcc[cnt_bcc].push_back(now.u);
}
if (bel[now.v] != cnt_bcc) {
bel[now.v] = cnt_bcc;
bcc[cnt_bcc].push_back(now.v);
}
} while (now.u != u || now.v != v);
fv(j, bcc[cnt_bcc]) in[bcc[cnt_bcc][j]] = 1;
if (dfs(bcc[cnt_bcc][0], 1))
fv(j, bcc[cnt_bcc]) vis[bcc[cnt_bcc][j]] = 1;
fv(j, bcc[cnt_bcc]) in[bcc[cnt_bcc][j]] = col[bcc[cnt_bcc][j]] = 0;
}
} else {
if (dfn[v] < dfn[u]) {
st.push(E[i]);
low[u] = min(low[u], dfn[v]);
}
}
}
}
}
using namespace Tarjan;
int main() {
freopen("input.txt", "r", stdin);
while (1) {
Read(n), Read(m);
if (!n && !m) return 0;
tot = 0;
fu(i, 1, n) head[i] = vis[i] = 0;
init();
fu(i, 1, n)
fu(j, 1, n)
g[i][j] = (i == j);
fu(i, 1, m) {
int u, v; Read(u), Read(v);
g[u][v] = g[v][u] = 1;
}
fu(i, 1, n)
fu(j, 1, n)
if (!g[i][j])
add(i, j);
fu(i, 1, n) if (!dfn[i]) tarjan(i, 0);
int ans = n;
fu(i, 1, n) if (vis[i]) --ans;
Write(ans), putchar('\n');
}
return 0;
}

POJ2942 Knights of the Round Table【Tarjan点双联通分量】【二分图染色】【补图】的更多相关文章

  1. 【POJ 2942】Knights of the Round Table(点双连通分量,二分图染色)

    圆桌会议必须满足:奇数个人参与,相邻的不能是敌人(敌人关系是无向边). 求无论如何都不能参加会议的骑士个数.只需求哪些骑士是可以参加的. 我们求原图的补图:只要不是敌人的两个人就连边. 在补图的一个奇 ...

  2. POJ2942 Knights of the Round Table(点双连通分量 + 二分图染色)

    题目大概说要让n个骑士坐成一圈,这一圈的人数要是奇数且大于2,此外有些骑士之间有仇恨不能坐在一起,问有多少个骑士不能入座. 双连通图上任意两点间都有两条不重复点的路径,即一个环.那么,把骑士看做点,相 ...

  3. POJ 2942 Knights of the Round Table 补图+tarjan求点双联通分量+二分图染色+debug

    题面还好,就不描述了 重点说题解: 由于仇恨关系不好处理,所以可以搞补图存不仇恨关系, 如果一个桌子上面的人能坐到一起,显然他们满足能构成一个环 所以跑点双联通分量 求点双联通分量我用的是向栈中pus ...

  4. UVA 1364 - Knights of the Round Table (获得双连接组件 + 二部图推理染色)

    尤其是不要谈了些什么,我想A这个问题! FML啊.....! 题意来自 kuangbin: 亚瑟王要在圆桌上召开骑士会议.为了不引发骑士之间的冲突. 而且可以让会议的议题有令人惬意的结果,每次开会前都 ...

  5. poj 2942 Knights of the Round Table - Tarjan

    Being a knight is a very attractive career: searching for the Holy Grail, saving damsels in distress ...

  6. 「题解」:[POJ2942]Knights of the Round Table

    问题 E: Knights of the Round Table 时间限制: 1 Sec  内存限制: 256 MB 题面 题目描述 作为一名骑士是一个非常有吸引力的职业:寻找圣杯,拯救遇难的少女,与 ...

  7. POJ2942 Knights of the Round Table[点双连通分量|二分图染色|补图]

    Knights of the Round Table Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 12439   Acce ...

  8. 【LA3523】 Knights of the Round Table (点双连通分量+染色问题?)

    Being a knight is a very attractive career: searching for the Holy Grail, saving damsels in distress ...

  9. Spoj 2878 KNIGHTS - Knights of the Round Table | 双联通分量 二分图判定

    题目链接 考虑建立原图的补图,即如果两个骑士不互相憎恨,就在他们之间连一条无向边. 显而易见的是,如果若干个骑士在同一个点数为奇数的环上时,他们就可以在一起开会.换句话说,如果一个骑士被一个奇环包含, ...

随机推荐

  1. C++类型前置声明

    前言 本文总结了c++中前置声明的写法及注意事项,列举了哪些情况可以用前置声明来降低编译依赖. 前置声明的概念 前置声明:(forward declaration), 跟普通的声明一样,就是个声明, ...

  2. 23TCP通信

    .pro 文件中加入:QT+=network 在Qt中实现TCP/IP服务器端通信的流程: 1.创建监听套接字,QTcpServer 2.将监听套接字设置为监听模式, listen 3.等待并接受客户 ...

  3. Python:导入numpy报错 No module named numpy

    Numpy是python的一种开源的数值计算扩展.这种工具可用来存储和处理大型矩阵,比python自身的嵌套列表结构要高效的多.但是在使用numpy时可能会出错(如上图). 解决办法:下载安装对应版本 ...

  4. Java IO的Reactor模式

    1.    Reactor出现的原因 Reator模式是大多数IO相关组件如Netty.Redis在使用时的IO模式,为什么需要这种模式,如何设计来解决高性能并发的呢? 最最原始的网络编程思路就是服务 ...

  5. 自动化运维工具SaltStack安装配置

    SaltStack是一种全新的基础设置管理方式,部署轻松,在几分钟内可运作起来,扩展性好,很容易管理上万台服务器,速度够快,服务器之间秒级通讯.通过部署SaltStack环境,我们可以在成千上万台服务 ...

  6. 图解Git命令【转】

    本文转载自:https://github.com/geeeeeeeeek/git-recipes/wiki/4.1-%E5%9B%BE%E8%A7%A3Git%E5%91%BD%E4%BB%A4 此页 ...

  7. HYSBZ 1036 树的统计Count(树链剖分)题解

    思路: 树链剖分,不知道说什么...我连模板都不会用 代码: #include<map> #include<ctime> #include<cmath> #incl ...

  8. jmeter 网速

    有人知道在jmeter 哪个里面哦 JMeterPlugins里面 network

  9. jenkins svn源码管理小记

    之前对Jenkins管理SVN源码的逻辑不太清楚,简单研究了一下.对于一般的开发通常会有以下三个环境: 1.svn服务器 2.Jenkins服务器 3.开发服务器(部署测试版web应用或者其他应用) ...

  10. install_github安装错误解决方法

    install.packages('devtools')library(devtools)install_github('hdng/clonevol') Installation failed: Ti ...