这道题看上去NPC啊,超级不可做的样子。

我们先分析一下简单的情形:没有\(x\)地图

此时每个地图由于限制掉一种汽车,那么显然只会有两种选择。

再考虑到限制的情况,那么大致做法就很显然了——2-SAT

首先是拆点,对于每张地图\(i\)拆成\(2i-1\)与\(2i\)表示这张地图选择能用的车的第一辆还是第二辆。

比如如果\(s_i=b\),那么\(2i-1\)表示选择\(A\)车,\(2i\)表示选择\(B\)车。

现在开始考虑选择的限制,对于每一个限制\((u,x,v,y)\),我们分情况讨论:

  • 当\(s_u=x\),说明这个限制没有意义,直接无视即可。
  • 当\(s_u\not=x\)且\(s_v=y\)时,说明这时\(u\)是绝对不能选\(x\)对应的点的。为了在建图中体现出来我们直接从\(x\)对应的点向另一个点连边即可。
  • 当\(s_u\not=x\)且\(s_v\not= y\)时,显然讲\(x\)对应的点向\(y\)对应的点连边。但是不要忘了2-SAT建图的对称性,我们还要从\(y\)对应的点的反点向\(x\)对应的点的反点连边。这样的意义也十分明确了吧,因为每个地图都必须做出选择,选择\(y\)对应的点的反点就使得\(u\)没有其他选择了。

然后就是用TarjanSCC的过程了,这里不再赘述。值得一提的是Tarjan的标号顺序是按照拓扑序的逆序(因为是DFS)跑出来的,所以可以直接用标号来输出方案。

那么接下来就是考虑万恶的\(x\)地图了

我们发现这样并没有什么好的建图方法,然后翻到下面一看数据范围\(d\le8\)

然后一种特别naive的想法就是\(3^d\)枚举每个\(x\)地图是什么类型。然后再乘上Tarjan的复杂度直接爆炸。

我们想一下我们枚举的本质:

  • 如果对于一个\(x\)地图取\(a\)即意味这只能用车\(B,C\)
  • 如果对于一个\(x\)地图取\(b\)即意味这只能用车\(A,C\)
  • 如果对于一个\(x\)地图取\(c\)即意味这只能用车\(A,B\)

然后我们惊奇地发现第三种情况已经被前两种情况包含掉了,所以复杂度变为\(2^d(n+m)\)

如果对于爆搜会被无解的数据卡满的同志们可以直接随机跑,实测这样也可以过。

不过实践的时候细节巨多,还是挺烦人的。

CODE

#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#define RI register int
#define Ms(f,x) memset(f,x,sizeof(f))
using namespace std;
const int N=50005;
struct ques
{
int x,y; char p1,p2;
}q[N<<1]; int n,m,d,tot,c[N],ukn[10],pt; char ch;
inline int T(char ch)
{
if (ch>='a'&&ch<='c') return ch-'a'+1;
else return ch-'A'+1;
}
inline int Id(int x,int y)
{
switch (x)
{
case 1:return y!=3?1:0;break;
case 2:return y!=3?1:0;break;
case 3:return y!=2?1:0;break;
}
}
inline char C(int x,int y)
{
switch (x)
{
case 1:return y!=2?'B':'C';break;
case 2:return y!=2?'A':'C';break;
case 3:return y!=2?'A':'B';break;
}
}
class FileInputOutput
{
private:
#define tc() (A==B&&(B=(A=Fin)+fread(Fin,1,S,stdin),A==B)?EOF:*A++)
#define S 1<<21
char Fin[S],Fout[S],*A,*B; int Ftop;
public:
FileInputOutput() { Ftop=0; A=B=Fin; }
inline void gc(char &ch)
{
while (!isalpha(ch=tc()));
}
inline void pc(char ch)
{
Ftop<S?Fout[Ftop++]=ch:(fwrite(Fout,1,S,stdout),Fout[(Ftop=0)++]=ch);
}
inline void read(int &x)
{
x=0; char ch; while (!isdigit(ch=tc()));
while (x=(x<<3)+(x<<1)+(ch&15),isdigit(ch=tc()));
}
inline void Fend(void)
{
fwrite(Fout,1,Ftop,stdout);
}
#undef S
#undef tc
}F;
class Two_SAT_Solver
{
private:
#define add(x,y) e[++cnt]=(edge){y,head[x]},head[x]=cnt
struct edge
{
int to,next;
}e[N<<2]; int head[N<<1],cnt,dfn[N<<1],low[N<<1],scc,col[N<<1],stack[N<<1],top,tot; bool vis[N];
inline int min(int a,int b)
{
return a<b?a:b;
}
#define to e[i].to
inline void Tarjan(int now)
{
dfn[now]=low[now]=++tot; stack[++top]=now; vis[now]=1;
for (RI i=head[now];i;i=e[i].next)
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])
{
col[now]=++scc; vis[now]=0; while (stack[top]!=now)
col[stack[top]]=scc,vis[stack[top--]]=0; --top;
}
}
#undef to
public:
inline void build(void)
{
for (RI i=1;i<=m;++i)
{
int x=q[i].x,y=q[i].y,p1=T(q[i].p1),p2=T(q[i].p2);
if (c[x]==p1) continue; int numx=x<<1,numy=y<<1;
if (c[y]==p2) add(numx-Id(c[x],p1),numx-(Id(c[x],p1)^1)); else
add(numx-Id(c[x],p1),numy-Id(c[y],p2)),add(numy-(Id(c[y],p2)^1),numx-(Id(c[x],p1)^1));
}
}
inline void check(void)
{
RI i; int lim=n<<1; for (i=1;i<=lim;++i) if (!dfn[i]) Tarjan(i);
for (i=1;i<=n;++i) if (col[(i<<1)-1]==col[i<<1]) return;
for (i=1;i<=n;++i) if (col[(i<<1)-1]<col[i<<1]) F.pc(C(c[i],1)); else F.pc(C(c[i],2));
F.Fend(); exit(0);
}
inline void clear(void)
{
cnt=tot=scc=0; Ms(head,0); Ms(dfn,0);
}
}S;
int main()
{
//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
RI i,j; for (F.read(n),F.read(d),i=1;i<=n;++i)
{
F.gc(ch); if (ch!='x') c[i]=T(ch); else ukn[++pt]=i;
}
for (F.read(m),i=1;i<=m;++i) F.read(q[i].x),F.gc(q[i].p1),F.read(q[i].y),F.gc(q[i].p2);
for (tot=(1<<d)-1,i=0;i<=tot;++i)
{
for (j=1;j<=d;++j) c[ukn[j]]=((i>>j-1)&1)+1;
S.build(); S.check(); S.clear();
}
return F.pc('-'),F.pc('1'),F.Fend(),0;
}

Luogu P3825 [NOI2017]游戏的更多相关文章

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

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

  2. P3825 [NOI2017]游戏

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

  3. Luogu 3825 [NOI2017]游戏

    Luogu的spj现在挂了,要去其他OJ提交. 2-SAT 发现如果不考虑$x$的情况,这就成为一个2-SAT的裸题了,我们有$O(n + m)$的方法可以解决它. 那加上$x$的情况怎么弄……岂不是 ...

  4. 洛谷P3825 [NOI2017]游戏(2-SAT)

    传送门 果然图论的题永远建图最麻烦……看着题解代码的建图过程真的很珂怕…… 先不考虑地图$x$,那么每一个地图都只能用两种赛车,于是我们可以用2-SAT来搞,用$i$表示这个地图能用的第一辆车,$i' ...

  5. 洛谷 P3825 [NOI2017]游戏 【2-SAT+状压】

    UOJ和洛谷上能A,bzoj 8ms即WA,现在也不是知道为啥--因为我太弱了 先看数据范围发现d非常小,自然想到了状压. 所以先假装都是只能跑两种车的,这显然就是个2-SAT问题了:对于x场没有hx ...

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

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

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

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

  8. [luogu]P1070 道路游戏[DP]

    [luogu]P1070 道路游戏 题目描述小新正在玩一个简单的电脑游戏.游戏中有一条环形马路,马路上有 n 个机器人工厂,两个相邻机器人工厂之间由一小段马路连接.小新以某个机器人工厂为起点,按顺时针 ...

  9. 题解 洛谷 P3825 【[NOI2017]游戏】

    从题面中四元组\((i,h_i,j,h_j)\)限制选择车子型号,不难想到这题要用\(2-SAT\)解决. 考虑转化为\(2-SAT\)模型,发现除地图\(x\)外,其他地图都只有两种车子型号可以参加 ...

随机推荐

  1. CentOS7 下源码安装 python3

    CentOS 7 下源码安装 python3   在CentOS7下,默认安装的是python2.7:为满足项目要求,安装python3 的方法如下:   1. 首先安装python3.6可能使用的依 ...

  2. 交叉编译器 arm-linux-gnueabi 和 arm-linux-gnueabihf 的区别

    交叉编译器 arm-linux-gnueabi 和 arm-linux-gnueabihf 的区别 自己之前一直没搞清楚这两个交叉编译器到底有什么问题,特意google一番,总结如下,希望能帮到道上和 ...

  3. Loadrunner 脚本录制-通过代理录制脚本

    脚本录制-通过代理录制脚本 by:授客 QQ:1033553122 版本:Loadruner 11.0 A.PC端录制Web应用程序 步骤1:根据实际情况,选择对应的协议 本例中选择Web(HTTP/ ...

  4. ​《数据库系统概念》1-数据抽象、模型及SQL

    ​DBMS(database-management system)包括数据库和用于存取数据的程序,DBMS的基本目标是为数据的存取提供方便.高效的方式,此外对大多数企业来说,数据是非常重要的,所以DB ...

  5. WRT 下 C++ wstring, string, String^ 互转

    由于项目原因,需要引入C++. wstring 与 string 的互转研究了一段时间,坑主要在于使用下面这种方式进行转换,中文会乱码 wstring ws = L"这是一段测试文字&quo ...

  6. Expo大作战(三十四)--expo sdk api之LinearGradient(线性渐变),KeepAwake(保持屏幕不休眠),IntentLauncherAndroid,Gyroscope,

    简要:本系列文章讲会对expo进行全面的介绍,本人从2017年6月份接触expo以来,对expo的研究断断续续,一路走来将近10个月,废话不多说,接下来你看到内容,讲全部来与官网 我猜去全部机翻+个人 ...

  7. 【Java入门提高篇】Day21 Java容器类详解(四)ArrayList源码分析

    今天要介绍的是List接口中最常用的实现类——ArrayList,本篇的源码分析基于JDK8,如果有不一致的地方,可先切换到JDK8后再进行操作. 本篇的内容主要包括这几块: 1.源码结构介绍 2.源 ...

  8. Window 由于未经处理的异常,进程终止。

    今天遇到了一个程序停止的问题: 应用程序: BussinessService.exe Framework 版本: v4.0.30319 说明: 由于未经处理的异常,进程终止.异常信息: System. ...

  9. MyBatis实现模糊查询的几种方式

    在学习MyBatis过程中想实现模糊查询,可惜失败了.后来上百度上查了一下,算是解决了.记录一下MyBatis实现模糊查询的几种方式. 数据库表名为test_student,初始化了几条记录,如图: ...

  10. P1056 排座椅

    非原创 #include<bits/stdc++.h>using namespace std;int t1[2009];int t2[2009]; int findmax(int *a){ ...