CF_400_D
题目大意:给出n扇门,m把钥匙,和没把钥匙可以改变状态(关->开,开->关》)的门的数量及对应编号(保证每个门被两把钥匙控制),现给出n扇门的初始状态(1表示开,0表示关),问是否可以通过这m把钥匙(可用可不用,不一定用完)使得所有的门都开启。
题解:终于看到了2-sat的能想出来的题了,2-sat本质是满族解的判断,这题显然可以用2-sat来求解,求解的一般套路是将每个点拆为两个点,分别表示取或者不取(即一件事的正反两面),最后通过2-sat算法,判断是否可行。这题需要一个转化,即我们对m拆点,表示用或者不用这把钥匙改变对应的门。那么如何连边?可以这样想:对于一个门 的初始状态,如果是开,那么两把钥匙要么都用,要么都不用;如果是关,一定是用一把弃一把。那么就是说连边是要用门的初始状态确定的,那么这题也就解决了。
#pragma comment(linker, "/STACK: 1024000000,1024000000")
#include <bits/stdc++.h>
#define pb push_back
#define mp make_pair
#define eb emplace_back
#define em emplace
#define pii pair<int,int>
#define de(x) cout << #x << " = " << x << endl
#define clr(a,b) memset(a,b,sizeof(a))
#define INF (0x3f3f3f3f)
#define LINF ((long long)(0x3f3f3f3f3f3f3f3f))
#define F first
#define S second
using namespace std;
inline int getint()
{
int _x=0; char _tc=getchar();
while(_tc<'0'||_tc>'9') _tc=getchar();
while(_tc>='0'&&_tc<='9') _x*=10,_x+=(_tc-'0'),_tc=getchar();
return _x;
}
const int N = 1e5 + 15;
int n, m;
int dor[N];
vector<int> key[N];
struct TwoSet
{
static const int MAXV = 100100;
int V, n, tp, S[MAXV<<1];
vector<int> g[MAXV<<1];
bool vis[MAXV<<1];
void init( int tt )
{
n = tt;
V = (tt<<1) + 1;
for ( int i = 1; i <= V; i ++ )
g[i].clear(), vis[i] = false;
}
void addEdge( int x, int y )
{
g[x].pb(y);
g[y].pb(x);
}
bool dfs( int x )
{
if ( vis[ x > n ? x - n : x + n ] ) return false;
if ( vis[x] ) return true;
vis[ S[++tp] = x ] = 1;
for ( int v : g[x] )
if ( !dfs(v) ) return false;
return true;
}
bool solve()
{
for ( int i = 1; i <= n; i ++ )
if ( !vis[i] && !vis[i+n] )
{
tp = 0;
if ( !dfs(i) )
{
while ( tp ) vis[S[tp--]] = false;
if ( !dfs(i+n) ) return false;
}
}
return true;
}
}TwoSet;
int main()
{
scanf("%d%d", &n, &m);
for ( int i = 1; i <= n; i ++ )
scanf("%d", &dor[i]);
for ( int i = 1, x; i <= m; i ++ )
{
scanf("%d", &x);
for ( int j = 0, y; j < x; j ++ )
scanf("%d", &y), key[y].pb(i);
}
TwoSet.init(m);
for ( int i = 1, x, y; i <= n; i ++ )
{
x = key[i][0], y = key[i][1];
if ( !dor[i] )
{
TwoSet.addEdge(x,y+m);
TwoSet.addEdge(x+m,y);
}
else
{
TwoSet.addEdge(x,y);
TwoSet.addEdge(x+m,y+m);
}
}
if ( TwoSet.solve() ) puts("YES");
else puts("NO");
return 0;
}
CF_400_D的更多相关文章
随机推荐
- SPOJ - DQUERY
题目链接:传送门 题目大意:一个容量 n 的数组, m次询问,每次询问 [x,y]内不同数的个数 题目思路:主席树(注意不是权值线段树而是位置线段树) 也就是按一般线段树的逻辑来写只是用主席树实现而已 ...
- 巡风代码架构简介以及Flask的项目文件结构简介
一.巡风: 巡风是一款什么东西,想必安全同行都不陌生吧.用它作为内网漏洞扫描管理架构是一种很好的选择,扫描快,开源,还可自己编写符合规则的POC直接放入相应目录来扩展.今天下午趁着有点时间捋了一下巡风 ...
- .net Asp AdRotator(广告控件)
1.新建项目名称AdRotator 2.右键项目名称添加一个xml文件命名为AdRotator.xml <?xml version="1.0" encoding=" ...
- 详探TextRange对象--查找与选择(转载)
TextRange对象是动态HTML(DHTML)的高级特性,使用它可以实现很多和文本有关的任务,例如搜索和选择文本.文本范围让您可以选择性的将字符.单词和句子从文档中挑选出来.TextRange对象 ...
- java 新手入门课程03
2017.7.6 java 课堂笔记 1.关于分支; if/else 是基于boolean 值的双分支 Switch 基于数字(包括整数 char byte 枚举, 字符串)类型的多分支 方法 ...
- Jenkins之构建执行脚本权限问题
Jenkins需要执行的脚本不在本机需要ssh免密码登陆到远程主机执行 Jenkins部署机ip地址为192.168.56.12 需要远程执行脚本的主机为192.168.56.11 设置好密钥可以使用 ...
- Spring boot官方文档学习(一)
个人说明:本文内容都是从为知笔记上复制过来的,样式难免走样,以后再修改吧.另外,本文可以看作官方文档的选择性的翻译(大部分),以及个人使用经验及问题. 其他说明:如果对Spring Boot没有概念, ...
- 以jar包的形式来使用前端的各种框架、组件。
springboot(二):web综合开发 - 纯洁的微笑博客 http://www.ityouknow.com/springboot/2016/02/03/spring-boot-web.html ...
- 寻找最小(最大)的k个数
题目描述:输入n个整数,输出其中最小的k个元素. 例如:输入1,2,3,4,5,6,7,8这8个数字,则最小的4个数字为1,2,3,4. 思路1:最容易想到的方法:先对这个序列从小到大排序,然后输出前 ...
- Select触发事件
案例1: <script type="text/JavaScript"> function gradeChange(){ var objS = document.ge ...