【LOJ6036】编码(2-sat)
【LOJ6036】编码(2-sat)
题面
题解
很显然的一个暴力:
枚举每个串中的?是什么,然后把和它有前缀关系的串全部给找出来,不合法的连边处理一下,那么直接跑\(2-sat\)就做完了。
现在的问题就在与不合法的数量可能会很多,所以需要优化这个连边的过程。
显然前缀关系和\(Trie\)树上的节点是相关的,即\(Trie\)树上这个串路径上所代表的所有点都是它的前缀,其子树中的每个串都以当前串为前缀。
首先构建\(Trie\),在往下的过程中对于路径上的所有已知前缀连边,这里肯定不能暴力一个个连,考虑把所有结尾的位置全部挂一条链挂在\(Trie\)树的这个节点上,那么只需要一路连下来就好了。
这样子就处理完了当前串的前缀。
但是当前串加入进来之后还可能会作为后面串的前缀,所以也要类似的挂在\(Trie\)树上。
连边的时候顺便处理好逆否命题的连边,这样子就不用建两次边了。
注意一下实际上这里需要两棵\(Trie\)树,一棵从上往下连,一棵从下往上连。
实际处理的时候把\(Trie\)树也建立正反两个点直接当做逆否命题连边就好了。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAX 3000300
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Line{int v,next;}e[MAX<<1];
int h[MAX],cnt=2;
inline void Add(int u,int v)
{
e[cnt]=(Line){v,h[u]};h[u]=cnt++;
u^=1;v^=1;
e[cnt]=(Line){u,h[v]};h[v]=cnt++;
}
int dfn[MAX],low[MAX],tim,G[MAX],scc;
int S[MAX],top;bool ins[MAX];
string s[MAX];
void Tarjan(int u)
{
dfn[u]=low[u]=++tim;ins[u]=true;S[++top]=u;
for(int i=h[u];i;i=e[i].next)
if(!dfn[e[i].v])Tarjan(e[i].v),low[u]=min(low[u],low[e[i].v]);
else if(ins[e[i].v])low[u]=min(low[u],dfn[e[i].v]);
if(dfn[u]==low[u])
{
++scc;int v;
do{v=S[top--];ins[v]=false;G[v]=scc;}while(u!=v);
}
}
int son[MAX][2],tot;
int n,len[MAX],p[MAX];
char ch[MAX];
void Insert(int u,int p)
{
int x=n+1,lst=0;
for(int i=0;i<len[u];++i)
{
if(!son[x][s[u][i]-48])son[x][s[u][i]-48]=++tot;
lst=x;x=son[x][s[u][i]-48];Add(p,x<<1);
}
int y=++tot;
Add(p,y<<1|1);Add(y<<1,x<<1);
son[lst][s[u][len[u]-1]-48]=y;
}
bool cmp(int a,int b){return len[a]<len[b];}
int main()
{
n=read();
for(int i=1;i<=n;++i)
{
scanf("%s",ch);len[i]=strlen(ch);p[i]=i;
for(int j=0;j<len[i];++j)s[i]+=ch[j];
}
sort(&p[1],&p[n+1],cmp);tot=n+1;
for(int i=1;i<=n;++i)
{
int u=p[i];
for(int j=0;j<len[u];++j)
if(s[u][j]=='?')
{
s[u][j]=48;Insert(u,u<<1);
s[u][j]=49;Insert(u,u<<1|1);
break;
}
else if(j==len[u]-1)Insert(u,u<<1),Add(u<<1|1,u<<1);
}
for(int i=1;i<=(tot<<1|1);++i)if(!dfn[i])Tarjan(i);
for(int i=1;i<=tot;++i)if(G[i<<1]==G[i<<1|1]){puts("NO");return 0;}
puts("YES");
return 0;
}
【LOJ6036】编码(2-sat)的更多相关文章
- LOJ6036 编码 2-SAT、Trie
传送门 每个串只有一个?,?还只能填0或者1,不难想到2-SAT求解. 一个很暴力的想法是枚举?填0或者1,然后对所有可能的前缀连边.这样边数是\(O(n^2)\)的,需要优化. 看到前缀不难想到Tr ...
- LOJ6036编码
每个串拆成两个,都插入trie数. 把trie树建出来后,每一条从根到叶子的链上最多只能有一个变量为1. 这是个经典的前后缀优化2-sat建图的套路. 树上的做法也就是边dfs边做而已. #inclu ...
- 前端笔记之服务器&Ajax(上)服务器&PHP&数据交互&HTTP
一.服务器 1.1 什么是服务器,做什么的? 服务器,就是放在机房中的电脑,和我们的电脑的区别在与服务器有固定的IP,服务器的安全性和稳定性相当的高;性能一般就可以了,但是CPU的性能要比普通的客户机 ...
- [LOJ6029~6052]雅礼集训 2017 选做
Link 代码可以在loj上看我的提交记录. Day 1 [LOJ6029]市场 对于一次除法操作,若区间内所有数的减少量均相同则可视作区间减法,否则暴力递归下去.显然一个线段树节点只会被暴力递归进去 ...
- 【LOJ6036】 「雅礼集训 2017 Day4」编码
传送门 LOJ Solution 因为?只有两种可能为0,1,所以就把这两个串搞出来. 那么现在?取0和?取1不能并存,前缀不能并存,所以就是一个\(2-SAT\),现在问题在于这个东西可能会有很多条 ...
- 前端学HTTP之实体和编码
前面的话 每天都有各种媒体对象经由HTTP传送,如图像.文本.影片以及软件程序等.HTTP要确保它的报文被正确传送,识别.提取以及适当处理.为了实现这些目标,HTTP使用了完善的标签来描述承载内容的实 ...
- 2.keras实现-->字符级或单词级的one-hot编码 VS 词嵌入
1. one-hot编码 # 字符集的one-hot编码 import string samples = ['zzh is a pig','he loves himself very much','p ...
- (转)x264 编码流程
转自:http://alphamailpost.blog.163.com/blog/static/20111808120128111160728/ http://www.usr.cc/thread-5 ...
- 服务器返回中文乱码的情况(UTF8编码 -> 转化为 SYSTEM_LOCALE 编码)
服务器乱码 转换使用如下方法 入惨{“msg”} -> utf8编码 -> 转化为 SYSTEM_LOCALE 编码 -> 接受转换后的参数 "sEncoding" ...
随机推荐
- oc之脚本
进入Build Phases页面,点击加号选择“New Run Script Phases”创建Run Script 在这里添加Run Script, 1.每次编译运行版本号(bundleVersio ...
- Python_阻塞IO、非阻塞IO、IO多路复用
0.承上 进程: 计算机里最小的资源分配单位: 数据隔离, 利用多核,数据不安全. 线程: 计算机中最小的CPU调度单位: 数据共享,GIL锁,数据不安全. 协程: 线程的一部分,是有用户来调度的; ...
- 福州大学软件工程1816 | W班 第3次作业成绩排名
写在前面 汇总成绩排名链接 1.作业链接 第三次作业--原型设计(结对第一次) 2.评分准则 本次作业总分 25分,由以下部分组成: (1)在随笔开头请加上该博客链接,以方便阅读时查看作业需求,并备注 ...
- JS 验证输入框输入 只允许输入正实数(正整数,正小数),其他情况下不能输入 oninput事件
input标签的oninput事件 要求输入框只输入正实数,包括整数和小数. 具体要求:整数部分不超过7位,可以没有小数,若有位数不超过2位. <input type="text&qu ...
- ES7的新特性
ES7的新特性 ES7 特性: 1.Array.prototype.includes2.Exponentiation Operator(求幂运算) 一,Array.prototype.includes ...
- AJAX+springmvc遇到的问题
当我使用AJAX将表单的值传入处理器中后,经过了一个判断再进行页面跳转时,不能在处理器中使用重定向,它会将重定向的页面内容在AJAX的data中显示出来而不是显示一个页面 所以只能在AJAX 的suc ...
- python之路--类的约束, 异常处理, MD5, 日志处理
一 . 类的约束 1. 写一个父类,父类中的某个方法要抛出一个异常 NotImplementedError class Base: # 对子类进行了约束. 必须重写该方法 # 以后上班了. 拿到公司代 ...
- synchronized与volatile的区别及各自的作用、原理(学习记录)
synchronized与volatile的区别,它们的作用及原理? 说到两者的区别,先要了解锁提供的两种特性:互斥(mutual exclusion) 和可见性(visibility). 互斥:即一 ...
- 使Python中的turtle模块画图两只小羊
turtle.circle(radius, extent=None, steps=None) 描述: 以给定半径画圆 参数: radius(半径); 半径为正(负),表示圆心在画笔的左边(右边)画圆 ...
- 三、kubernetes环境搭建(实践)
一.目前近况 docker 版本 K8S支持 18.06的 二.安装docker #1.配置仓库 sudo yum install -y yum-utils device-mapper-persist ...