codeforces_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的更多相关文章

随机推荐

  1. 微信开发(2):微信js sdk分享朋友圈,朋友,获取config接口注入权限验证(转)

    进行微信开发已经一阵子了,从最初的什么也不懂,到微信授权登录,分享,更改底部菜单,素材管理,等. 今天记录一下微信jssdk 的分享给朋友的功能,获取config接口注入. 官方文档走一下简单说:四步 ...

  2. [shell]用shell脚本将本地文件夹与ftp上的文件夹同步

    需求说明 最近在AIX上做开发,开发机器在office网段,测试机器在lab网段,不能互相通讯,只能通过特定的ftp来传文件. 每次上传的机器都要做:登录ftp,进入我的目录,上传:下载的机器都要做: ...

  3. zabbix 部署 jmx 监控tomcat

    zabbix提供了一个java gateway的应用去监控jmx(Java Management Extensions,即Java管理扩展)是一个为应用程序.设备.系统等植入管理功能的框架.JMX可以 ...

  4. SaltStack部署redis主从

    需求: 一,部署redis主从,一台主一台从 二,redis监听自己的IP地址,而不是0.0.0.0 主:安装,配置,启动 从:安装,配置,启动,主从

  5. HDU 1103 Flo's Restaurant(模拟+优先队列)

    Flo's Restaurant Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

  6. MySQL 慢查询日志工具之pt-query-digest

    1. 工具简介 pt-query-digest是用于分析mysql慢查询的一个工具,它可以分析binlog.General log.slowlog,也可以通过SHOWPROCESSLIST或者通过tc ...

  7. Puppet学习:Augeas的怪问题

    今天测试部署Zabbix,本来以前是正常的,今天莫名其妙报错.测试后发现是Augeas出现问题,无论如何都不执行.于是采用调试模式: puppet agent --test --verbose --d ...

  8. 【Python】接口自动化测试-Fidder的使用(未完待续……)

    一.fidder一些一定需要掌握的知识. 1.工具简介 2.清屏操作(1中提到了,这里再着重说明下): 3.get和post请求参数相关: 4.会话框(Fidder左侧区域内容解析): 5.Reque ...

  9. 整合最优雅SSM框架:SpringMVC + Spring + MyBatis 基础

    在写代码之前我们先了解一下这三个框架分别是干什么的? 相信大以前也看过不少这些概念,我这就用大白话来讲,如果之前有了解过可以跳过这一大段,直接看代码! SpringMVC:它用于web层,相当于con ...

  10. mysqldump 导出统一限制每张数据表导出的记录数

    mysqldump 导出统一限制每张数据表导出的记录数 在工作过程中,需要将生产的数据导出到本地开发环境,我希望可以导出部分数据.而服务器数据量比较大(上千万),如果选择直接从服务器导出数据, 正在运 ...