【NOI2017】游戏 题解(2-SAT+缩点)
题目大意:有四种场地$a,b,c,x$和三种赛车$A,B,C$,$a$不能跑$A$,$b$不能跑$B$,$c$不能跑$C$,$x$都可以跑。给定$n$个场地和$m$个四元组$(i,h_i,j,h_j)$,意为如果在第$i$个场地跑$h_i$,那么第$j$个场地必须跑$h_j$。输出一种合法方案。若无输出$-1$。
----------------------------------
此题难在建图。把建图搞定就是一个裸的2-SAT板子。
我们规定$0$为每种场地的第一种能跑的车(按字典序),$1$表示第二种能跑的车。对于约束条件,我们有$3$种情况:
1.$h_i$类型的车与$i$冲突,那么我们直接跳过,不用管它。
2.$h_j$类型的车与$j$冲突,那么我们让$i$向$i'$连边,表示如果选择$h_i$则无解。
3.没有冲突情况。那么我们直接按照2-SAT建图方式建就好,即$i$向$j$连边,$j'$向$i'$连边。
对于$x$,我们只要把它当作某一种特定的场地,然后$dfs$枚举所有情况就好。把它当作任意两种场地就可以包含所有可能情况。
时间复杂度$O(2^dn)$。
PS:一开始我超级傻逼,写了一串$if$来判各种情况还各种出锅;后面突然想到只要写个函数判断就行了草。从早上7点来学校写,一直到上午10点才过……
代码:
#include<bits/stdc++.h>
using namespace std;
const int maxn=;
int dfn[maxn],low[maxn],vis[maxn],pos[maxn],jishu,tot;
stack<int> st;
int head[maxn],cnt,n,m,d;
int query[maxn][];
char ch[maxn][],c[maxn],s[maxn];
struct node
{
int next,to;
}edge[maxn];
inline int read()
{
int x=,f=;char ch=getchar();
while(!isdigit(ch)){if (ch=='-') f=-;ch=getchar();}
while(isdigit(ch)){x=x*+ch-'';ch=getchar();}
return x*f;
}
inline void add(int from,int to)
{
edge[++cnt].next=head[from];
edge[cnt].to=to;
head[from]=cnt;
}
inline void clear()
{
jishu=tot=cnt=;
memset(head,,sizeof(head));
memset(dfn,,sizeof(dfn));
memset(low,,sizeof(low));
memset(pos,,sizeof(pos));
while(!st.empty()) st.pop();
}
inline void tarjan(int now)
{
dfn[now]=low[now]=++jishu;
st.push(now);vis[now]=;
for (int i=head[now];i;i=edge[i].next)
{
int to=edge[i].to;
if (!dfn[to]) tarjan(to),low[now]=min(low[now],low[to]);
else if (vis[to]) low[now]=min(low[now],dfn[to]);
}
if (low[now]==dfn[now])
{
tot++;
while(st.top()!=now)
{
int x=st.top();st.pop();
vis[x]=;
pos[x]=tot;
}
int x=st.top();st.pop();
vis[x]=;
pos[x]=tot;
}
}
inline bool check(int pos,char x)
{
if (x+==s[pos]) return ;
return ;
}
inline bool get(int pos,char x)
{
if (s[pos]=='a') return x=='B'?:;
else return x=='A'?:;
}
inline void dfs(int now)
{
if (now>n)
{
clear();
for (int i=;i<=m;i++)
{
int a=query[i][],b=query[i][];
if (check(a,ch[i][])) continue;
if (check(b,ch[i][]))
{
int flag=get(a,ch[i][]);
if (!flag) add(a,a+n);
else add(a+n,a);
continue;
}
else
{
int x=get(a,ch[i][]),y=get(b,ch[i][]);
if (x==&&y==) add(a,b),add(b+n,a+n);
if (x==&&y==) add(a,b+n),add(b,a+n);
if (x==&&y==) add(a+n,b),add(b+n,a);
if (x==&&y==) add(a+n,b+n),add(b,a);
}
}
for (int i=;i<=*n;i++) if(!dfn[i]) tarjan(i);
for (int i=;i<=n;i++) if (pos[i]==pos[i+n]) return;
for (int i=;i<=n;i++)
{
if (pos[i]<pos[i+n])
{
if (s[i]=='a') printf("B");
else printf("A");
}
else
{
if (s[i]=='c') printf("B");
else printf("C");
}
}
exit();
}
if (c[now]=='a') s[now]='a',dfs(now+);
if (c[now]=='b') s[now]='b',dfs(now+);
if (c[now]=='c') s[now]='c',dfs(now+);
if (c[now]=='x'){s[now]='a',dfs(now+);s[now]='b',dfs(now+);}
}
int main()
{
n=read(),d=read();
cin>>(c+);
m=read();
for (int i=;i<=m;i++)
{
query[i][]=read(),cin>>ch[i][];
query[i][]=read(),cin>>ch[i][];
}
dfs();
printf("-1");
return ;
}
【NOI2017】游戏 题解(2-SAT+缩点)的更多相关文章
- BZOJ4945 & 洛谷3825 & UOJ317:[NOI2017]游戏——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=4945 https://www.luogu.org/problemnew/show/P3825 ht ...
- 【BZOJ4945】[Noi2017]游戏 2-SAT
[BZOJ4945][Noi2017]游戏 题目描述 题解:2-SAT学艺不精啊! 这题一打眼看上去是个3-SAT?哎?3-SAT不是NPC吗?哎?这题x怎么只有8个?暴力走起! 因为x要么不是A要么 ...
- P3825 [NOI2017]游戏
题目 P3825 [NOI2017]游戏 做法 \(x\)地图外的地图好做,模型:\((x,y)\)必须同时选\(x \rightarrow y,y^\prime \rightarrow x^\pri ...
- [Luogu P3825] [NOI2017] 游戏 (2-SAT)
[Luogu P3825] [NOI2017] 游戏 (2-SAT) 题面 题面较长,略 分析 看到这些约束,应该想到这是类似2-SAT的问题.但是x地图很麻烦,因为k-SAT问题在k>2的时候 ...
- [UOJ317]【NOI2017】游戏 题解
题意 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 A.B.C 表示.地图一共有四种,分别用小写字 ...
- 【题解】NOI2017游戏
2-SAT.洛谷P3845 一开始以为——怎么有3个呢?后来发现因为每个地图都有一种车是不能用的,所以就等于每一个地图都有两个适应的车啦. 那么对于x类型的地图呢——只有8个,直接2^8暴力枚举每一种 ...
- 题解 洛谷 P3825 【[NOI2017]游戏】
从题面中四元组\((i,h_i,j,h_j)\)限制选择车子型号,不难想到这题要用\(2-SAT\)解决. 考虑转化为\(2-SAT\)模型,发现除地图\(x\)外,其他地图都只有两种车子型号可以参加 ...
- bzoj3825 NOI2017 游戏
题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行nn 场游戏,每场游戏使用一张地 ...
- NOI2017 [NOI2017]游戏 【2-sat】
题目 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行nn 场游戏,每场游戏使用 ...
随机推荐
- Yii2源码分析(一):入口
写在前面,写这些随笔是记录下自己看Yii2源码的过程,可能会有些流水账,大部分解析放在注释里说明,由于个人水平有限,有不正确的地方还望斧正. web入口文件Index.php // 定义全局的常量,Y ...
- mongodb(三):数据库连接(python)
import pymongo def get_mongodb_conn(**kwargs): db_host = kwargs.get('host') db_port = kwargs.get('po ...
- hihoCoder 1050 树中的最长路 最详细的解题报告
题目来源:树中的最长路 解题思路:枚举每一个点作为转折点t,求出以t为根节点的子树中的‘最长路’以及与‘最长路’不重合的‘次长路’,用这两条路的长度之和去更新答案,最终的答案就是这棵树的最长路长度.只 ...
- Vue中使用websocket
<template> <div class="test">websocket demo</div></template> < ...
- Ethical Hacking - POST EXPLOITATION(1)
METERPRETER BASICS >help - shows help >background - backgrounds current session >sessions - ...
- 集训作业 洛谷P1101 单词方阵
这个题的长度真的有点长,我直接放图片吧 这个题又是一个和谐的搜索,找到yizhong的y就开始8面搜索,遇见正确的字母就继续搜索,不正确就果断放弃,果然又是一个和谐的搜索呢. #include< ...
- Nginx/Httpd负载均衡tomcat配置
在前一篇博客中我们聊了下用Nginx和httpd对后端tomcat服务做反代相关配置,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/13334180.html: ...
- [jvm] -- 内存模型篇
内存模型 JDK1.6 JDK1.8 线程私有的: 程序计数器 虚拟机栈 本地方法栈 线程共享的: 堆 方法区 直接内存 (非运行时数据区的一部分) 程序计数器 线程私有 两个作用 字节码解释器通 ...
- consul++ansible+shell批量下发注册node_exporter
--日期:2020年7月21日 --作者:飞翔的小胖猪 文档功能说明: 文档通过ansible+shell+consul的方式实现批量下发安装Linux操作系统监控的node_exporter软件, ...
- Oracle可视化工具连接
Oracle可是化工具有很多,以下只列举sql developer和sql plus这两款连接方式 sql developer: SQL Develope启动后,需要创建一个数据库连接,只有创建了数据 ...