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

输出一组合法的选数方案,无解输出$-1$

题解:考虑$d=0$的情况,就是$2-sat$的裸题。

那$d>0$的呢?发现$d\leqslant8$,因为跑一次$2-sat$的复杂度是$O(n+m)(n\leqslant5\times10^4,m\leqslant10^5)$,好像有很大的空间乱搞?若暴力$dfs$每一位选什么,复杂度是$O(3^8(n+m))\approx9.8\times10^8$,过不了。

考虑优化,如果枚举$d$个位置不可以填什么,那么就是$1,2;2,3;1,3$,发现前两种已经包含了$1,2,3$三种方案,于是复杂度成了$O(2^8(n+m))\approx3.8\times10^7$。可以承受,于是就过了

卡点:1.限制条件给的是大写字母,写成小写字母

​   2.有一个函数因为调试改成了另一个,最后忘记调回来,花了我$1.5h+$的时间

C++ Code:

#include <cstdio>
#include <cstring>
#include <iostream>
#define maxn 50010
#define maxm 100010
int n, nn, D, m, tot;
int X[10], p[maxn];
int a[maxm], b[maxm], c[maxm], d[maxm];
char s[maxn]; int head[maxn << 1], cnt;
struct Edge {
int to, nxt;
} e[maxm << 1];
void addE(int a, int b) {
e[++cnt] = (Edge) {b, head[a]}; head[a] = cnt;
} int DFN[maxn << 1], low[maxn << 1], idx;
int S[maxn << 1], top, res[maxn << 1], CNT;
bool ins[maxn << 1];
inline int min(int a, int b) {return a < b ? a : b;}
void tarjan(int x) {
DFN[x] = low[x] = ++idx; ins[S[++top] = x] = 1;
int y;
for (int i = head[x]; i; i = e[i].nxt)
if (!DFN[y = e[i].to]) tarjan(y), low[x] = std::min(low[x], low[y]);
else if (ins[y]) low[x] = std::min(low[x], DFN[y]);
if (DFN[x] == low[x]) {
CNT++;
do ins[y = S[top--]] = 0, res[y] = CNT; while (x != y);
}
} void init() {
memset(head, 0, sizeof head), cnt = 0;
memset(DFN, 0, sizeof DFN), idx = 0;
CNT = 0;
}
inline bool get(int a, int b) {return b != (a + 1) % 3;}
inline char reget(int a, int b) {return (b + a + 1) % 3 + 'A';}
inline int P(int a, int b) {return a << 1 | b;}
bool work(int T) {
init();
for (int i = 0; i < D; i++) p[X[i]] = bool(T & 1 << i);
for (int i = 0; i < m; i++) {
if (p[a[i]] != b[i]) {
if (p[c[i]] == d[i]) addE(a[i] << 1 | get(p[a[i]], b[i]), a[i] << 1 | !get(p[a[i]], b[i]));
else {
addE(a[i] << 1 | get(p[a[i]], b[i]), c[i] << 1 | get(p[c[i]], d[i])),
addE(c[i] << 1 | !get(p[c[i]], d[i]), a[i] << 1 | !get(p[a[i]], b[i]));
}
}
}
for (int i = 2; i <= nn + 1; i++) {
if (!DFN[i]) tarjan(i);
if (i & 1) if (res[i] == res[i - 1]) return false;
}
for (int i = 1; i <= n; i++) printf("%c", reget(res[i << 1] > res[i << 1 | 1], p[i]));
return true;
} int main() {
scanf("%d%d%s", &n, &D, s + 1); nn = n << 1;
for (int i = 1; i <= n; i++) {
if (s[i] == 'x') X[tot++] = i;
p[i] = s[i] - 'a';
}
scanf("%d", &m);
for (int i = 0; i < m; i++) {
char B, D;
scanf("%d %c %d %c", a + i, &B, c + i, &D);
b[i] = B - 'A';
d[i] = D - 'A';
}
int U = 1 << D;
for (int i = 0; i < U; i++) if (work(i)) return 0;
printf("-1");
return 0;
}

  

[bzoj4945][Noi2017]游戏的更多相关文章

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

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

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

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

  3. P3825 [NOI2017]游戏

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

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

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

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

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

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

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

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

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

  8. bzoj3825 NOI2017 游戏

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

  9. [NOI2017]游戏(2-SAT)

    这是约半年前写的题解了,就搬过来吧 感觉这是NOI2017最水的一题(当然我还是不会2333),因为是一道裸的2-SAT.我就是看着这道题学的2-SAT 算法一:暴力枚举.对于abc二进制枚举,对于x ...

随机推荐

  1. 微信小程序禁止下拉_解决小程序下拉出现空白的情况

    微信小程序禁止下拉 在微信小程序中,用力往下拉动,页面顶部会出现一段空白的地方. 产品的需求不太允许这么做,会影响用户体验,查看文档发现可以使用enablePullDownRefresh这属性来实现, ...

  2. #Python编程从入门到实践#第四章笔记

    #Python编程从入门到实践#第四章笔记   操作列表 ​​​1.遍历列表 使用for循环,遍历values列表 for value in values: print(value) 2.数字列表 使 ...

  3. 笔记--tslib 编译

    tslib 是qt启动时的一个触屏校正检验程序. 它的配置以及编译比较简单. 第一步, 下载tslib源码包: http://download.csdn.net/detail/MKNDG/329156 ...

  4. Matplotlib 图表的基本参数设置

    1.图名,图例,轴标签,轴边界,轴刻度,轴刻度标签 # 图名,图例,轴标签,轴边界,轴刻度,轴刻度标签等 df = pd.DataFrame(np.random.rand(10,2),columns= ...

  5. [BZOJ2243][SDOI2011]染色(树链剖分)

    [传送门] 树链剖分就行了,注意线段树上颜色的合并 Code #include <cstdio> #include <algorithm> #define N 100010 # ...

  6. Kings(状压DP)

    Description 用字符矩阵来表示一个8x8的棋盘,'.'表示是空格,'P'表示人质,'K'表示骑士.每一步,骑士可以移动到他周围的8个方格中的任意一格.如果你移动到的格子中有人质(即'P'), ...

  7. 笔记-scrapy-scarpyd

    笔记-scrapy-scarpyd 1.      scrapy部署 会写爬虫之后就是部署.管理爬虫了,下面讲一下如何部署scrapy爬虫. 现在使用较多的管理工具是Scrapyd. scrapyd是 ...

  8. 2 js的20/80关键知识

    1. 2 var a = 1; undefined a 1 alert(a); undefined var b = true; var c = "Hi"; undefined al ...

  9. IIS 部署网站--浏览--“该页无法显示”

    解决办法: 打开IIS管理器--web站点(网站)--右键点击对应的站点--属性--主目录--执行权限改为(脚本和和执行文件) 点击“应用”--确定. 重启一下站点,OK.

  10. Pascal游戏 大富翁MUD

    大富翁MUD Pascal源码 Chaobs改编自百度网友作品 此源码非Chaobs原创,但转载时请仍注明出处. 估计90后以后就没有孩子知道MUD了. program wxtw; uses crt; ...