【NOI2017】游戏(搜索,2-sat)

题面

BZOJ的SPJ是假的

兹磁洛谷

题解

如果没有\(x\)地图的影响

这就是一个裸的\(2-sat\)问题

但是现在有不超过\(8\)个\(x\)地图的影响

我们不难想到枚举\(x\)地图的状态再来\(2-sat\)判断剩余是否可行。

这样的复杂度是\(O(3^dn)\),稍微算一下发现这个复杂度有点假

考虑如何优化,我们枚举\(x\)地图不连什么

表面上看起来还是\(O(3^dn)\)

但是,当他等价于\(a,b\)两种地图时,\(2^d\)种方案中一定会包含着最后的解,也就是说,\(a,b\)两种地图替换\(x\)的所有状态,等价于\(a,b,c\)三种地图替换\(x\)

这样的复杂度是\(O(2^dn)\)

现在考虑\(2-sat\)如何连边

对于限制\(i->j\)

显然直接连接\(i->j\)

并且,如果\(j'\)被选择,那么必须选择\(i'\)

所以,同时连接\(j'->i'\)

一些没有意义的连边可以直接省略。

至于输出方案,\(Tarjan\)求出来的强连通分量已经和拓扑序相关,因此没有必要重新建图拓扑排序

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 333333
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
char g[MAX];
struct Limit{int i,j;char a,b;}q[MAX];
int n,D,m,pos[20];
struct Line{int v,next;}e[MAX<<3];
int h[MAX],cnt;
int p[MAX];
inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;}
bool fl;
void Build()
{
memset(h,0,sizeof(h));cnt=1;fl=true;
for(int i=1;i<=n;++i)
{
p[i]=p[i+n]=p[i+n+n]=0;
if(g[i]=='a')p[i+n]=i+n+n,p[i+n+n]=i+n;
else if(g[i]=='b')p[i]=i+n+n,p[i+n+n]=i;
else p[i]=i+n,p[i+n]=i;
}
for(int i=1;i<=m;++i)
{
int u=q[i].i,v=q[i].j;
int a=q[i].a-65,b=q[i].b-65;
if(u==v&&a==b)continue;
if(g[u]-97==a)continue;
if(g[v]-97==b){Add(u+a*n,p[u+a*n]);continue;}
Add(u+a*n,v+b*n);
Add(p[v+b*n],p[u+a*n]);
}
}
int dfn[MAX],low[MAX],S[MAX],top;
int gr,G[MAX],tim,ans[MAX];
bool Ins[MAX];
void Tarjan(int u)
{
dfn[u]=low[u]=++tim;
S[++top]=u;Ins[u]=true;
int v;
for(int i=h[u];i;i=e[i].next)
{
v=e[i].v;
if(!dfn[v])Tarjan(v),low[u]=min(low[u],low[v]);
else if(Ins[v])low[u]=min(low[u],dfn[v]);
}
if(dfn[u]==low[u])
{
++gr;
do{v=S[top--];G[v]=gr;Ins[v]=false;}while(u!=v);
}
}
void Calc()
{
Build();gr=tim=0;if(!fl)return;
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(G,0,sizeof(G));
for(int i=1;i<=n+n+n;++i)if(!dfn[i])Tarjan(i);
for(int i=1;i<=n+n+n;++i)if(G[i]==G[p[i]])return;
for(int i=1;i<=n;++i)
{
if(g[i]=='a')(G[i+n]<G[i+n+n])?ans[i]=1:ans[i]=2;
if(g[i]=='b')(G[i]<G[i+n+n])?ans[i]=0:ans[i]=2;
if(g[i]=='c')(G[i]<G[i+n])?ans[i]=0:ans[i]=1;
}
for(int i=1;i<=n;++i)putchar(ans[i]+65);puts("");
exit(0);
}
void dfs(int x)
{
if(x==D+1){Calc();return;}
g[pos[x]]='a';dfs(x+1);
g[pos[x]]='b';dfs(x+1);
}
int main()
{
n=read();D=read();D=0;
scanf("%s",g+1);
m=read();
for(int i=1;i<=m;++i)
q[i].i=read(),q[i].a=getchar(),q[i].j=read(),q[i].b=getchar();
for(int i=1;i<=n;++i)
if(g[i]=='x')pos[++D]=i;
dfs(1);
puts("-1");
return 0;
}

【BZOJ4945】【NOI2017】游戏(搜索,2-sat)的更多相关文章

  1. BZOJ 4945 NOI2017 游戏 搜索+2-SAT

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=4945 分析: 首先考虑没有x的情况,发现有一个明显的推理模型,容易看出来可以用2-SAT ...

  2. [BZOJ4945][Noi2017]游戏 2-sat

    对于所有的x,我们枚举他的地图类型,事实上我们只需要枚举前两种地形就可以覆盖所有的情况. 之后就变成了裸的2-sat问题. 对于一个限制,我们分类讨论: 1.h[u]不可选,跳过 2.h[v]不可选, ...

  3. [bzoj4945][Noi2017]游戏

    题目大意:有$n$个位置,有三种数,每个位置只可以填一种数,$d(d\leqslant8)$个位置有三种选择,其他位置只有两种选择.有一些限制,表示第$i$个位置选了某种数,那么第$j$个位置就只能选 ...

  4. 【BZOJ4945】[Noi2017]游戏 2-SAT

    [BZOJ4945][Noi2017]游戏 题目描述 题解:2-SAT学艺不精啊! 这题一打眼看上去是个3-SAT?哎?3-SAT不是NPC吗?哎?这题x怎么只有8个?暴力走起! 因为x要么不是A要么 ...

  5. P3825 [NOI2017]游戏

    题目 P3825 [NOI2017]游戏 做法 \(x\)地图外的地图好做,模型:\((x,y)\)必须同时选\(x \rightarrow y,y^\prime \rightarrow x^\pri ...

  6. [Luogu P3825] [NOI2017] 游戏 (2-SAT)

    [Luogu P3825] [NOI2017] 游戏 (2-SAT) 题面 题面较长,略 分析 看到这些约束,应该想到这是类似2-SAT的问题.但是x地图很麻烦,因为k-SAT问题在k>2的时候 ...

  7. 【bzoj4945】[Noi2017]游戏(搜索+2-sat)

    bzoj 洛谷 题意: 现在有\(a,b,c\)三种车,每个赛道可能会存在限制:\(a\)表示不能选择\(a\)类型的赛车,\(b,c\)同理:\(x\)表示该赛道不受限制,但\(x\)类型的个数$\ ...

  8. 并不对劲的bzoj4945:loj2305:uoj317:p3825[NOI2017]游戏

    题目大意 2-SAT,其中有\(d\)(\(d\leq 8\))个点是\(3-SAT\). 题解 枚举\(d\)个点不取三个中(假设三个为\(a,b,c\))的哪一个,然后整体变成做\(2-SAT\) ...

  9. BZOJ4945 & 洛谷3825 & UOJ317:[NOI2017]游戏——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4945 https://www.luogu.org/problemnew/show/P3825 ht ...

  10. NOIp 2011 mayan游戏 搜索

    题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...

随机推荐

  1. 使用 AFNetworking做过断点续传吗?

    断点续传的主要思路: 检查服务器文件信息 检查本地文件 如果比服务器文件小, 断点续传, 利用 HTTP 请求头的 content-range实现断点续传(如果content-range不存在就取Co ...

  2. C#随堂

    顺序语句 上到下执行 分支语句 if    else switch() { case 1: Console.WriteLine(1); break; case 2: Console.WriteLine ...

  3. EF6+MVC5之Oracleo数据库的CodeFirst方式实现

    http://www.oracle.com/technetwork/topics/dotnet/whatsnew/index.htm http://www.knowsky.com/887470.htm ...

  4. python全栈开发- 前⽅⾼能-迭代器

    python_day_12 今日主要内容 1, 函数名的应用,第一类对象 函数名可以像变量一样进行使用 1.赋值 2.作为list元素 3.作为参数 4.作为返回值 2, 闭包 内部函数访问外部函数的 ...

  5. 车牌,车架号,VIN码毫秒识别技术,汽车后市场的春天到来了

    vin码(车架号)识别运用 不仅在制造.销售.保养.保险.车辆评估.交易环节会需要录入汽车的VIN码,在交通事故处理中,作为汽车身份唯一识别码,VIN码是处理事故的执法人员必须要记录的信息之一.随着汽 ...

  6. python--自定义模块

    python模块说明:类似于函数式编程和面向过程编程,函数式编程则完成一个功能,其他代码用来调用即可,提供了代码的重用性和代码间的耦合.而对于一个复杂的功能来,可能需要多个函数才能完成(函数又可以在不 ...

  7. redis 批量删除操作

    redis 批量删除操作 需要在redis里面清空一批数据,redis没有支持通配符删除, 只有del key1 key2 ... 但是可以通配符获取 KEYS PATTERN 然后利用linux管道 ...

  8. 如何使用phpredis连接Redis的方法

    本文跟大家介绍使用同一VPC内弹性云服务器ECS上的phpredis连接Redis的方法. 更多的客户端的使用方法,请参考https://redis.io/clients 前提条件 已成功申请Redi ...

  9. 算法笔记(c++)--使用一个辅助栈排列另一个栈

    算法笔记(c++)--使用一个辅助栈排列另一个栈 仅仅使用一个辅助栈,不使用其他数据结构来排列一个栈,要求,上大下小. 分析下.肯定是先吧主栈中的数据都放到辅助栈中,在辅助栈中上小下大. 1.首先循环 ...

  10. Live Archive 训练题 2019/3/9

    7454 Parentheses A bracket is a punctuation mark, which is used in matched pairs, usually used withi ...