【LOJ#6036】[雅礼集训2017Day4]编码
题意简述
判定 n 个含 ? 字符的二进制串是否存在一种把 0/1 填入 ? 中的方案使得任意两个串不具有前缀关系。
(一个串最多一个 ?)
Sol
二进制串 ,并且一个串最多一个 '?'
很容易想到用 2-sat 和 trie 树。
那么问题变为插入这些二进制串,在 ‘?’ 处选择向哪边插入 ,使得任意一个结束节点的祖先节点中不存在一个结束节点。
然后我们考虑构建 2-sat 模型,首先每一个串分配一个变量表示 '?' 选择了什么(没有?就随便强制选则某一个就行了)
当然之后我们一种思路是直接用这些变量来判定并解决问题 ,不过复杂度显然是 \(O(n^2)\) 的。
考虑优化,因为是祖先中不能存在结束节点,我们给每一个 trie 树上的节点加一个变量表示除去自己外的祖先中是否存在结束节点,这个东西显然从上到下具有传递性,那么可以初步建图了。
然后看怎么体现一个串的 '?' 的决策情况。发现这个东西不是很好做,因为结束节点不会受到自己影响。那么我们直接新建一个变量好了,表示包含了当前位置的情况下祖先是否有结束节点。
之后我们还需要解决一个问题,就是一些不同的串可能共用了结束位置,我们显然不能够让他们共用一个变量,因为他们之间也是互相影响的,并且他们还来自不同的 '?' ,不能共用。
发现这个东西可以直接和上面的情况合并到一个变量上,因为本来我们插入完之后就要新建一个点,那么正好一起用,也不影响答案,把该连的边连上就行了。
code:
#include<bits/stdc++.h>
using namespace std;
template<class T>inline void init(T&x){
x=0;char ch=getchar();bool t=0;
for(;ch>'9'||ch<'0';ch=getchar()) if(ch=='-') t=1;
for(;ch>='0'&&ch<='9';ch=getchar()) x=(x<<1)+(x<<3)+(ch-48);
if(t) x=-x;return;
}
#define TR(a) ((a)<<1|1)
#define FA(a) ((a)<<1)
const int N=5e5+10;
const int MAXN=3e6+10;
int son[MAXN][2];
char S[N];
char *s[N];int len[N],id[N];
int n,cnt=0;
inline bool cmp(int i,int j){return len[i]<len[j];}
struct edge{
int to,next;
}a[MAXN];
int head[MAXN],cur=0,dfn[MAXN],low[MAXN],bel[MAXN],stk[MAXN],top=0,vis[MAXN],I=0,bcc;
inline void add(int x,int y){a[++cur]=(edge){y,head[x]};head[x]=cur;}
void tarjan(int u){
dfn[u]=low[u]=++I,stk[++top]=u,vis[u]=1;
for(int v,i=head[u];i;i=a[i].next){
v=a[i].to;
if(!dfn[v]) tarjan(v),low[u]=min(low[u],low[v]);
else if(vis[v]) low[u]=min(low[u],dfn[v]);
}
if(dfn[u]==low[u]) {
int v;++bcc;
do{v=stk[top--];vis[v]=0;bel[v]=bcc;if(bel[v]==bel[v^1]) {puts("NO"),exit(0);}}while(v!=u);
}
return;
}
void Insert(char*s,int len,int Jud){
int p=0,las=0;
for(int i=1;i<=len;++i) {
int c=s[i]-'0';
las=p;
if(son[p][c]) p=son[p][c];
else {
son[p][c]=++cnt;
if(p) add(TR(p),TR(cnt)),add(FA(cnt),FA(p));
p=cnt;
}
}
++cnt;//新建点以防止出现多个串共用了一个点的情况 , 新建的点是要考虑当前点的!
add(Jud,FA(p)),add(TR(p),Jud^1);
add(TR(p),TR(cnt)),add(FA(cnt),FA(p));
add(Jud,TR(cnt)),add(FA(cnt),Jud^1);
son[las][(int)(son[las][1]==p)]=cnt;
}
int main()
{
init(n);int now=1;cnt=n;// n 个串选择什么
for(int i=1;i<=n;++i){
scanf("%s",S+now);
len[i]=strlen(S+now);
s[i]=&S[now-1];now+=len[i];
id[i]=i;
}sort(id+1,id+1+n,cmp);
for(int i=1;i<=n;++i) {
int t=id[i];int j=0;
for(j=1;j<=len[t];++j) {
if(s[t][j]=='?') {// 有问号
s[t][j]='0';Insert(s[t],len[t],FA(t));//填 0
s[t][j]='1';Insert(s[t],len[t],TR(t));//填 1
break;
}
}
if(j>len[t]) {add(TR(t),FA(t));Insert(s[t],len[t],FA(t));}
}
for(int i=0;i<=cnt;++i) if(!dfn[i]) tarjan(i);
puts("YES");
return 0;
}
【LOJ#6036】[雅礼集训2017Day4]编码的更多相关文章
- LOJ.6504.[雅礼集训2018 Day5]Convex(回滚莫队)
LOJ 莫队.发现只需要维护前驱后继就可以了. 但是加入一个点需要找到它当前的前驱后继,很麻烦还带个\(\log\). 但是如果只有删除某个点,只需要更新一下它的前驱后继即可. 用回滚莫队就好惹. 撤 ...
- Loj#6503-「雅礼集训 2018 Day4」Magic【分治NTT】
正题 题目链接:https://loj.ac/p/6503 题目大意 \(n\)张卡\(m\)种,第\(i\)种卡有\(a_i\)张,求所有排列中有\(k\)对相邻且相同的卡牌. \(1\leq n\ ...
- $loj\ 6045$ [雅礼集训 $2017\ Day8$] 价 网络流
正解:网络流 解题报告: 传送门$QwQ$ 这题还,挺有趣的我$jio$得. 考虑依然先是照着最小割的模子建图呗,然后从意义上来分析,割一条边就相当于不吃一种减肥药/买一种药材.由已知得,买的药材数量 ...
- [LOJ 6031]「雅礼集训 2017 Day1」字符串
[LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...
- [LOJ 6030]「雅礼集训 2017 Day1」矩阵
[LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...
- [LOJ 6029]「雅礼集训 2017 Day1」市场
[LOJ 6029] 「雅礼集训 2017 Day1」市场 题意 给定一个长度为 \(n\) 的数列(从 \(0\) 开始标号), 要求执行 \(q\) 次操作, 每次操作为如下四种操作之一: 1 l ...
- Loj #6503. 「雅礼集训 2018 Day4」Magic
Loj #6503. 「雅礼集训 2018 Day4」Magic 题目描述 前进!前进!不择手段地前进!--托马斯 · 维德 魔法纪元元年. 1453 年 5 月 3 日 16 时,高维碎片接触地球. ...
- loj #6046. 「雅礼集训 2017 Day8」爷
#6046. 「雅礼集训 2017 Day8」爷 题目描述 如果你对山口丁和 G&P 没有兴趣,可以无视题目背景,因为你估计看不懂 …… 在第 63 回战车道全国高中生大赛中,军神西住美穗带领 ...
- LOJ_6045_「雅礼集训 2017 Day8」价 _最小割
LOJ_6045_「雅礼集训 2017 Day8」价 _最小割 描述: 有$n$种减肥药,$n$种药材,每种减肥药有一些对应的药材和一个收益. 假设选择吃下$K$种减肥药,那么需要这$K$种减肥药包含 ...
随机推荐
- FreeBSD上安装Cassandra 3.10
哈哈,你居然点进来了,来吧,一起吐槽FreeBSD啊,装了一上午Cassandra 3.10都没有装成功, 终于,鄙人一条 shutdown -p now 结束了FreeBSD,默默打开了CentOS ...
- String对象的常见方法
String 对象方法 方法 描述 anchor() 创建 HTML 锚. big() 用大号字体显示字符串. blink() 显示闪动字符串. bold() 使用粗体显示字符串. charAt() ...
- 阶段3 1.Mybatis_05.使用Mybatis完成CRUD_8 Mybatis中的返回值深入-调整实体类属性解决增和改方法的报错
如果实体类和数据库表内的字段没有保持一致 修改了属性子弹. 重新生成getter和setter还有toString方法 测试类里面,相关set的方法就都会报错. 只需要修改为最新的set方法就可以了. ...
- robotframework之常用系统关键字
常用系统关键字此处做些记录,后续根据实际应用陆续补充 1.变量声明 ${a} Set Variable hello 2.表单嵌套 Select Frame Xpath=//* [@] Unselect ...
- Linx下Keepalived做成服务
在/usr目录下面执行: find -name keepalived 返回如下: ./sbin/keepalived ./local/sbin/keepalived ./local/etc/keepa ...
- centos7 主从dns配置 bind服务
一,配置前请先关闭防火墙selinux 防火墙关闭方法,参见上一篇文章. setenforce 0 #临时关闭 修改/etc/selinux/config 文件 将SELINUX=enforc ...
- Spring Security Session Time Out
最近在用Spring Security做登录管理,登陆成功后,页面长时间无操作,超过session的有效期后,再次点击页面操作,页面无反应,需重新登录后才可正常使用系统. 为了优化用户体验,使得在se ...
- Linux-Maven部署
一.Maven是什么 二.Maven部署 1.环境信息: (1)centos7.3 (2)jdk1.8 (3)maven3.5.3 2.安装jdk (1)下载地址[http://www.oracle. ...
- Win10使用自带功能创建系统映像备份时D盘被包含进去问题的解决
在使用Windows10系统时,使用Windows自带功能创建系统映像备份文件时碰到了一些问题,所以在此记录一下. 创建系统映像文件的步骤,如下: 1.打开 控制面板 -> 选择 系统和安全 - ...
- 数塔 Medium
Summer is coming! It's time for Iahub and Iahubina to work out, as they both want to look hot at the ...