问题描述

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. 阿里iconfont的使用

    1.找到阿里巴巴图标库 2.找到图标 3.搜索你想要的图标 4.将图标添加到购物车 5.点击右上角的购物车按钮,我这里添加了两个. 6.提示你登陆,不需要花钱,找其中一个账号登陆一下就行了 假如你使用 ...

  2. mysql-5.7.27安装

    1,下载5.7.27安装包   百度网盘   (注:5.7.27要安装net faframework4.5.2) 2,创建my.ini及data文件 下载5.7.27后解压,在创建my.ini文件及d ...

  3. Visual Studio 开发(三):Visual Studio 使用时常见问题解决方案

    一.Error LNK2019: 无法解析的外部符号 此问题应该是Visual Studio的初学者最常碰到的问题,也是相对来说很让人头疼的问题. 注:Error LNK2019 问题在VC 6.0 ...

  4. 基于Moya、RxSwift和ObjectMapper优雅实现REST API请求

    在Android开发中有非常强大的 Retrofit 请求,结合RxJava可以非常方便实现 RESTful API 网络请求.在 iOS开发中也有非常强大的网络请求库 Moya ,Moya是一个基于 ...

  5. MySQL数据库:合并结果集

    合并结果集 union----合并结果集 对合并后的结果集中的重复数据也会自动去重 select sName from students union select tName from Teacher ...

  6. 闲话复数(1) | 不现实的虚数 i 为什么虚?它长成什么样?

    原文 | https://mp.weixin.qq.com/s/y-Nb3S508UZuf_0GtRuNaQ 复数的英文是complex number,直译是复杂的数.最早接触复数大概是在高中时期,只 ...

  7. shell中字典的一个用法示例

    1. shell中字典的用法示例, 数组用法同字典 2. 以上使用sell配合字典实现的功能, 使用awk能轻松搞定, 如下: awk '{print $2}' file.txt | sort | u ...

  8. 第04组 Beta冲刺(3/4)

    队名:斗地组 组长博客:地址 作业博客:Beta冲刺(3/4) 各组员情况 林涛(组长) 过去两天完成了哪些任务: 1.分配展示任务 2.收集各个组员的进度 3.写博客 展示GitHub当日代码/文档 ...

  9. SpringBoot系列之日志框架使用教程

    目录 1.SpringBoot日志级别 1).日志级别简介 2).默认日志级别 3).配置日志级别 4).日志分组设置 2.SpringBoot日志格式设置 1).默认格式原理简介 2).默认日志格式 ...

  10. C++入门到理解阶段二基础篇(8)——C++指针

    1.什么是指针? 为了更加清楚的了解什么是指针?我们首先看下变量和内存的关系,当我们定义了int a=10之后.相当于在内存之中找了块4个字节大小的空间,并且存储10,要想操作这块空间,就通过a这个变 ...