问题描述

LG3825

BZOJ4945

LOJ2305


题解

发现对于每个地图,如果没有\(A,B,C\)地图不可以使用\(a,b,c\),就是一个\(\mathrm{3-SAT}\)问题。

有了这个限制之后,\(A,B,C\)地图就变为了\(\mathrm{2-SAT}\)问题,但是\(x\)地图还是\(\mathrm{3-SAT}\)

因为\(\mathrm{k-SAT}(3 \le k)\)是一个\(\mathrm{NP}\)完全问题,观察到\(d \le 8\),于是直接爆搜即可。

爆搜出每个\(x\)的状态,\(\mathrm{2-SAT}\)即可。


\(\mathrm{Code}\)

#include<bits/stdc++.h>
using namespace std; template <typename Tp>
void read(Tp &x){
x=0;char ch=1;int fh;
while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
if(ch=='-'){
fh=-1;ch=getchar();
}
else fh=1;
while(ch>='0'&&ch<='9'){
x=(x<<1)+(x<<3)+ch-'0';
ch=getchar();
}
x*=fh;
} const int maxn=100000+7;
const int maxm=200000+7; int n,m,d; int Head[maxn],Next[maxm],to[maxm],tot;
int pic[maxn],all[maxn],cot; struct node{
int x,xx,y,yy;
}limt[maxm]; void add(int x,int y){
to[++tot]=y,Next[tot]=Head[x],Head[x]=tot;
} void fr_contest(int &x){
char ch=1;
while(ch!='x'&&ch!='a'&&ch!='b'&&ch!='c') ch=getchar();
if(ch=='x') x=4;
else if(ch=='a') x=1;
else if(ch=='b') x=2;
else x=3;
} void fr_limit(int &x){
char ch=1;
while(ch!='A'&&ch!='B'&&ch!='C') ch=getchar();
if(ch=='A') x=1;
else if(ch=='B') x=2;
else x=3;
} bool ins[maxn];
int sta[maxn],top,bel[maxn],cnt;
int dfn[maxn],low[maxn],ind; void tarjan(int x){
dfn[x]=low[x]=++ind;ins[x]=1,sta[++top]=x;
for(int i=Head[x];i;i=Next[i]){
int y=to[i];
if(dfn[y]){
if(ins[y]) low[x]=min(low[x],dfn[y]);
}
else{
tarjan(y);
low[x]=min(low[x],low[y]);
}
}
if(dfn[x]==low[x]){
++cnt;
while(sta[top]!=x){
bel[sta[top]]=cnt,ins[sta[top]]=0,top--;
}
bel[x]=cnt,ins[x]=0,top--;
}
} int id[maxn][4]; int opp(int x){
if(x>n) return x-n;
return x+n;
} void solve(){
memset(Head,0,sizeof(Head));memset(Next,0,sizeof(Next));
memset(dfn,0,sizeof(dfn));memset(low,0,sizeof(low));
tot=top=cnt=ind=0;memset(ins,0,sizeof(ins));memset(bel,0,sizeof(bel));
for(int i=1;i<=m;i++){
if(pic[limt[i].x]==limt[i].xx) continue;
if(pic[limt[i].y]==limt[i].yy){
add(id[limt[i].x][limt[i].xx],opp(id[limt[i].x][limt[i].xx]));
}
else{
add(id[limt[i].x][limt[i].xx],id[limt[i].y][limt[i].yy]);
add(opp(id[limt[i].y][limt[i].yy]),opp(id[limt[i].x][limt[i].xx]));
}
}
for(int i=1;i<=2*n;i++){
if(!dfn[i]) tarjan(i);
}
// for(int i=1;i<=n;i++){
// printf("node %d:%d %d %d\n",i,id[i][1],id[i][2],id[i][3]);
// }
for(int i=1;i<=n;i++){
if(bel[i]==bel[i+n]) return;
}
for(int i=1;i<=n;i++){
if(bel[i]<bel[i+n]){
if(pic[i]==1) printf("B");
else printf("A");
}
else{
if(pic[i]==1||pic[i]==2) printf("C");
else printf("B");
}
}
exit(0);
} void dfs(int step){
if(step==d+1){
solve();return;
}
int k=all[step];
pic[k]=1,id[k][2]=k,id[k][3]=k+n;
dfs(step+1);
pic[k]=2,id[k][1]=k,id[k][3]=k+n;
dfs(step+1);
pic[k]=4;
} int main(){
#ifndef ONLINE_JUDGE
freopen("game.in","r",stdin);
#endif
read(n);read(d);
for(int i=1;i<=n;i++){
fr_contest(pic[i]);
if(pic[i]==4){
all[++cot]=i;
}
if(pic[i]==1){
id[i][2]=i,id[i][3]=i+n;
}
else if(pic[i]==2){
id[i][1]=i,id[i][3]=i+n;
}
else if(pic[i]==3){
id[i][1]=i,id[i][2]=i+n;
}
}
read(m);
for(int i=1;i<=m;i++){
read(limt[i].x);fr_limit(limt[i].xx);
read(limt[i].y);fr_limit(limt[i].yy);
}
dfs(1);
printf("-1");
return 0;
}

LG3825/BZOJ4945/LOJ2305 「NOI2017」游戏 dfs+2-SAT的更多相关文章

  1. LOJ2305 「NOI2017」游戏

    「NOI2017」游戏 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行$n$场 ...

  2. 「NOI2017」游戏

    「NOI2017」游戏 题目描述 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 \(A\).\(B\).\ ...

  3. loj #2305. 「NOI2017」游戏

    #2305. 「NOI2017」游戏 题目描述 小 L 计划进行 nnn 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 AAA.BBB. ...

  4. LOJ_2305_「NOI2017」游戏 _2-sat

    LOJ_2305_「NOI2017」游戏 _2-sat 题意: 给你一个长度为n的字符串S,其中第i个字符为a表示第i个地图只能用B,C两种赛车,为b表示第i个地图只能用A,C两种赛车,为c表示第i个 ...

  5. 「NOI2017」游戏 解题报告

    「NOI2017」游戏 \(d\)这么小,你考虑直接对\(d\)个东西暴力 枚举\(x\)为\(a\)或\(b\)(\(c\)就不用了,因为\(a,b\)已经包含\(c\))了,剩下的就是个\(2-s ...

  6. 【LOJ】 #2305. 「NOI2017」游戏

    题解 枚举x所在的地图的颜色,然后2-SAT建边 如果v所在的地图刚好是不能选的,那么u这边只能选另一种颜色 否则就是u的颜色到v的颜色 v的另一种颜色到u的另一种颜色 代码 #include < ...

  7. loj#2305. 「NOI2017」游戏 2-sat

    链接 https://loj.ac/problem/2305 https://www.luogu.org/problemnew/show/P3825 思路 3-sat神马的就不要想了,NP问题 除去x ...

  8. 「HNOI2018」游戏

    「HNOI2018」游戏 解题思路 首先没有锁上的门可以缩点缩掉,然后对于一扇锁上的门,如果钥匙在左边,那么右边就永远不可能到达左边,同理如果钥匙在右边,左边就永远不可能到达右边. 然后考虑一个暴力的 ...

  9. LOJ2303 「NOI2017」蚯蚓排队

    「NOI2017」蚯蚓排队 题目描述 蚯蚓幼儿园有$n$只蚯蚓.幼儿园园长神刀手为了管理方便,时常让这些蚯蚓们列队表演. 所有蚯蚓用从$1$到$n$的连续正整数编号.每只蚯蚓的长度可以用一个正整数表示 ...

随机推荐

  1. 【Puppeteer】puppeteer安装/常用的方法以及一个小栗子(Youtube油管自动评论)

    这里介绍的是Win平台的安装方法,其他平台请至Github>Puppeteer. 首先要安装node.js 可以看我这篇的开头>[Angular]学习笔记-环境部署.项目建立相关 1.新建 ...

  2. 通过程序调用微信公众号发消息api返回48001

    自己的订阅号,尝试通过写程序来给用户发消息.结果呢,接口返回报错:errcode=48001,errmsg = api unauthorized hint: [ZlPULa02942276!] 去微信 ...

  3. 快速搭建用于测试的rtsp协议网络流媒体数据服务

    背景: 最近根据项目需求,在平台系统中加入了视频监控显示功能,但是限于没有提供真实可用的监控摄像头数据,通过EasyScreenLive快速搭建了一个rtsp的流媒体服务,下面将实现步骤分享给大家,为 ...

  4. Node.js Error简介以及捕获方式

    error的类型nodejs 的error 一般分为四种类型: 1.标准的 JavaScript 错误,例如 EvalError.SyntaxError.RangeError.ReferenceErr ...

  5. orm终极大爆炸

    orm终极 甩一个代码给你 # 创建字段 class Field: def __init__(self, name, column_type, primary_key, default): self. ...

  6. TCP/IP网络协议初识

    目录 一.什么是协议? 二.什么是TCP/IP协议? 三.TCP/IP为什么这么多协议? 四.TCP/IP协议为什么分层? 五.TCP/IP协议如何入门? 六.TCP/IP 的分层: 七.各协议层打包 ...

  7. sql server编写脚本求解第1天1分钱之后每天两倍持续一个月的等比数列问题

    一.问题 问题1 场景:如果你未来的丈母娘要求你,第1天给她1分钱,第2天给2分钱,第3天给4分钱,以此类推,每天给前一天的2倍,给1个月(按30天)算就行.问:第30天给多少钱,总共给多少钱? 问题 ...

  8. LuaSocket 学习笔记

    --- LUA SocketLib 和 协程 前言: 这是一篇译文(The LUA SocketLib and the Coroutines),有删改,原文见下方链接. 简介 目标读者:会使用 LUA ...

  9. python连接sqlite3

    一.了解sqlite sqlite是一种嵌入式数据库,它的数据库就是一个文件.由于SQLite本身是用C写的,而且体积很小,所以经常被集成到各种应用程序中,甚至在IOS和Android的APP中都可以 ...

  10. 一篇文章看懂angularjs component组件

     壹 ❀ 引 我在 angularjs 一篇文章看懂自定义指令directive 一文中详细介绍了directive基本用法与完整属性介绍.directive是个很神奇的存在,你可以不设置templa ...