传送门

分析:

并查集:

第一步先将所有矛盾从大至小排序,显然先将矛盾值大的分成两部分会更优。

普通的并查集都只能快速合并两个元素至同一集合,却不能将两个元素分至不同集合。

对于将很多数分成两个集合,并给出两数存在的矛盾关系(A和B不能在一集合),普通并查集无法解决。

考虑见每个元素拆成两个点,拆出来的点是它的对立点(i+n),如果要把A和B分至不同集合,就连边 A->B+n,B->A+n(假设先不压缩路径),也就是说:将A与B的对立点分至一个集合 \(\Leftrightarrow\) 将A与B分至不同集合。当然这种思路只能解决分成两部分的情况。

二分+二分图染色

其实就是要求把有矛盾的分到不同组里,于是可以使用二分图染色。先排序,然后二分不能满足的第一个位置,将前面的矛盾关系进行二分图染色,如果然不成功(相邻节点颜色相同),那么更新答案。注意二分图染色要对每个未染色的点都染一次,因为建出来的图可能不联通。

code

并查集:

#include<bits/stdc++.h>
using namespace std;
const int N = 2e4 + 50, M = 1e5 + 50;
int n, m, anc[N * 2];
struct node{
int x, y, w;
inline bool operator < (const node &b) const{
return w > b.w;
}
}edge[M];
inline int getAnc(int x){return x == anc[x] ? x : (anc[x] = getAnc(anc[x]));}
int main(){
freopen("h.in", "r", stdin);
scanf("%d%d", &n, &m);
for(int i = 1; i <= 2 * n; i++) anc[i] = i;
for(int i = 1; i <= m; i++)scanf("%d%d%d", &edge[i].x, &edge[i].y, &edge[i].w);
sort(edge + 1, edge + m + 1);
for(int i = 1; i <= m; i++){
int fx = getAnc(edge[i].x), fy = getAnc(edge[i].y);
if(fx == fy){
printf("%d\n", edge[i].w);
return 0;
}
else{
anc[fx] = getAnc(edge[i].y + n);
anc[fy] = getAnc(edge[i].x + n);
}
}
printf("0");
return 0;
}

二分+二分图染色:

#include<bits/stdc++.h>
using namespace std;
const int N = 2e4 + 40, M = 1e5 + 50;
int n, m, col[N];
vector<int> G[N];
struct State {
int u, c, f;
State() {}
State(int _u, int _c, int _f):u(_u), c(_c), f(_f) {}
};
queue<State> que;
struct node {
int a, b, c;
inline bool operator < (const node &g) const {
return c > g.c;
}
} item[M]; inline bool BiCheck(int mid) {
memset(col, -1, sizeof(int) * (n + 5));
while(!que.empty()) que.pop();
for(int i = 1; i <= n; i++) G[i].clear();
for(int i = 1; i <= mid; i++) {
G[item[i].a].push_back(item[i].b);
G[item[i].b].push_back(item[i].a);
} for(int i = 1; i <= n; i++){
if(col[i] == -1){
col[i] = 0;
que.push(State(i, 0, 0));
while(!que.empty()) {
State t = que.front();
que.pop();
int u = t.u, c = t.c, f = t.f;
for(int e = G[u].size() - 1; e >= 0; e--) {
int v = G[u][e];
if(v == f) continue;
if(col[v] != -1) {
if(col[v] == c) return false;
else continue;
} else col[v] = c ^ 1, que.push(State(v, c ^ 1, u));
}
}
} }
return true;
} int main() {
freopen("h.in", "r", stdin);
scanf("%d%d", &n, &m);
for(int i = 1; i <= m; i++) scanf("%d%d%d", &item[i].a, &item[i].b, &item[i].c);
sort(item + 1, item + m + 1); int l = 1, r = m, ans = 0;
while(l <= r) {
int mid = l + r >> 1;
if(!BiCheck(mid)) ans = item[mid].c, r = mid - 1;
else l = mid + 1;
}
printf("%d", ans);
return 0;
}

LUOGU 1525 关押罪犯 - 并查集拆点(对立点) / 二分+二分图染色的更多相关文章

  1. NOIP 2010 关押罪犯 并查集 二分+二分图染色

    题目描述: S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用"怨气值" ...

  2. Luogu P1525 [NOIp2010提高组]关押罪犯 | 并查集

    题目链接 这一道题,我用了并查集来做.在此题中,并查集的作用就是:将同一个监狱里的罪犯合并到一起. 思路:将每对罪犯之间的怨气值从大到小排序,再依次把他们分到不同的两个监狱里,当发现这一对罪犯已经在同 ...

  3. NOIP2010关押罪犯[并查集|二分答案+二分图染色 | 种类并查集]

    题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用“怨气值”(一个正整数值)来表示 ...

  4. 洛谷P1525关押罪犯——并查集

    题目:https://www.luogu.org/problemnew/show/P1525 并查集+贪心,从大到小排序,将二人分在不同房间,找到第一个不满足的即为答案. 代码如下: #include ...

  5. [noip2010]关押罪犯 并查集

    第一次看的时候想到了并查集,但是不知道怎么实现: 标解,f[i]表示i所属的集合,用f[i+n]表示i所属集合的补集,实现的很巧妙,可以当成一个使用并查集的巧妙应用: #include<iost ...

  6. NOIP2010提高组] CODEVS 1069 关押罪犯(并查集)

    这道这么简单的题目还写了这么久.. 将每个会发生冲突的两人的怒气进行排序,然后从怒气大到小,将两个人放到不同监狱中.假如两人都已经被放置且在同一监狱,这就是答案. ------------------ ...

  7. P1525 关押罪犯 并查集

    题目描述 SS城现有两座监狱,一共关押着NN名罪犯,编号分别为1-N1−N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用“怨气值”(一个正整数值) ...

  8. luogu1525 [NOIp2011]关押罪犯 (并查集)

    先从大到小排序,看到哪个的时候安排不开了 给每个人拆成两个,如果x和y有矛盾,就给x和y‘.y和x’连边:如果a和b(或a'和b')在同一个集合里,说明他们一定要在同一个监狱里. #include&l ...

  9. 关押罪犯 - 并查集&优先队列

    题目地址:http://www.51cpc.com/web/problem.php?id=4261 Summarize: 此题最巧妙的是“敌人的敌人就是朋友!”,故需先将敌对关系放入优先队列,按怨恨值 ...

随机推荐

  1. System.out.println 的多线程并发问题

    假设println函数的參数为常量则不会出现线程并发问题,可是假设參数为表达式形式.则JVM在运行println函数的时候会分为几步来运行,从而造成并发问题. 例如以下样例所看到的: package ...

  2. 在C#中实现Word页眉页脚的全部功能

    页眉页脚经常使用于文章排版,在Word工具栏里.我们能够加入页眉,页脚,页码,日期和时间.图片等信息和内容.页眉/页脚有两个额外选项:首页不同,奇偶页不同.有时在不同的节(section)里插入不同的 ...

  3. Maven搭建Spring Security3.2项目详解

    本来是打算在上一篇SpringMVC+Hibernate上写的,结果发现上面那篇 一起整合的,结果发现上一篇内容实在是太长了,就另起一篇,这篇主要是采用 Maven搭建Spring+SpringMVC ...

  4. 11.3 Android显示系统框架_最简单的surface测试程序

    APP有一个surface(界面),其有多个buffer用来存放界面数据,这些buffer是向surfaceflinger申请的: 因此我们编写的surface测试程序步骤: (1)获得surface ...

  5. python3 登录验证小程序,同一用户输错三次密码,锁定账户

    ''' 让用户输入用户名密码 认证成功后显示欢迎信息用户3次认证失败后,退出程序,再次启动程序尝试登录时,还是锁定状态''' # !/usr/bin/env python # -*- coding:u ...

  6. QSYS系统简介

    QSYS是SoPC Builder的新一代产品. QSYS系统集成工具自动生成互联逻辑,连接IP和子系统 QSYS的设计理念是提高设计抽象级,从而使机器自动生成底层代码. Altera的Avalon总 ...

  7. SDWebImage使用

    SDWebImage这个类库提供一个UIImageView类别以支持加载来自网络的远程图片.具有缓存管理.异步下载.同一个URL下载次数控制和优化等特征. 将SDWebImage类库添加入工程时,一定 ...

  8. UITextField用法

    .创建 .UITextField* textField = [[UITextField alloc]initWithFrame:CGRectMake(, , , )]; .设置委托 //委托类需要遵守 ...

  9. 30、在LCD上显示摄像头图像

    1. 准备虚拟机2.安装工具链sudo tar xjf arm-linux-gcc-4.3.2.tar.bz2 -C /设置环境变量:sudo vi /etc/environment : PATH=& ...

  10. 如何调试Javascript代码

    转自原文如何调试Javascript代码 目前,常用的浏览器IE.Chrome.Firefox都有相应的脚本调试功能.作为我们.NET 阵营,学会如何在IE中调试JS就足够了,在掌握了IE中的调试方法 ...