CF582E Boolean Function(DP,状态压缩,FMT)
简单题。
我第二道自己做出来的 2900
没毛病,我没切过 2800 的题
lqy:“CF 评分 2800 是中等难度”
我活个啥劲啊
为了方便(同时压缩状态个数),先建出表达式树,然后一棵子树就代表一个完整的表达式(要么是单个变量,要么是被一堆匹配的括号恰好包住的)。
注意:(在我的写法里面)叶子都是变量,非叶子都是运算符,每个非叶子有恰好两个儿子,就表示这个运算符把整个表达式分成了哪两个,
然后就开始 DP 了。
明显状压吧?首先四个变量的取值可以压成一个四位二进制数(\(0\) 到 \(15\))。然后再状压一次,压成一个 \(16\) 位二进制数,第 \(i\) 位表示四个变量取值情况是 \(i\) 时表达式的值。
\(dp[u][i]\) 就表示 \(u\) 为根的子树,然后)&@!$^(&*!@#为 \(i\) 的方案数。
暴力转移是 \(O(|s|4^{16})\),不太行。
发现转移是个与卷积和或卷积的形式,FMT 优化。
时间复杂度是 \(O(|s|\times 16\times 2^{16})\),虽然看起来不太稳但是跑得飞快(400ms)……
#include<bits/stdc++.h>
using namespace std;
const int maxn=5555,mod=1000000007,lim=65536;
#define lson o<<1,l,mid
#define rson o<<1|1,mid+1,r
#define FOR(i,a,b) for(int i=(a);i<=(b);i++)
#define ROF(i,a,b) for(int i=(a);i>=(b);i--)
#define MEM(x,v) memset(x,v,sizeof(x))
inline int read(){
int x=0,f=0;char ch=getchar();
while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
int n,dp[255][lim],m,a[16],b[16],with[maxn],stk[maxn],tp,cnt,ch[maxn][2],fa[maxn],id[maxn],rt,p[lim],q[lim];
char s[maxn],val[maxn];
bool op[maxn];
void FMT_pre(int *A){
for(int i=1;i<lim;i<<=1)
for(int j=0;j<lim;j+=i<<1)
FOR(k,0,i-1) A[j+k]=(A[j+k]+A[i+j+k])%mod;
}
void IFMT_pre(int *A){
for(int i=1;i<lim;i<<=1)
for(int j=0;j<lim;j+=i<<1)
FOR(k,0,i-1) A[j+k]=(A[j+k]-A[i+j+k]+mod)%mod;
}
void FMT_suf(int *A){
for(int i=1;i<lim;i<<=1)
for(int j=0;j<lim;j+=i<<1)
FOR(k,0,i-1) A[i+j+k]=(A[i+j+k]+A[j+k])%mod;
}
void IFMT_suf(int *A){
for(int i=1;i<lim;i<<=1)
for(int j=0;j<lim;j+=i<<1)
FOR(k,0,i-1) A[i+j+k]=(A[i+j+k]-A[j+k]+mod)%mod;
}
void dfs(int now){
if(ch[now][0]) dfs(ch[now][0]);
if(ch[now][1]) dfs(ch[now][1]);
if(!op[now]){
switch(val[now]){
case 'A':dp[now][65280]=1;break;
case 'B':dp[now][61680]=1;break;
case 'C':dp[now][52428]=1;break;
case 'D':dp[now][43690]=1;break;
case 'a':dp[now][255]=1;break;
case 'b':dp[now][3855]=1;break;
case 'c':dp[now][13107]=1;break;
case 'd':dp[now][21845]=1;break;
case '?':
dp[now][65280]=dp[now][61680]=dp[now][52428]=dp[now][43690]=
dp[now][255]=dp[now][3855]=dp[now][13107]=dp[now][21845]=1;
}
}
else{
if(val[now]!='|'){
FOR(i,0,lim-1) p[i]=dp[ch[now][0]][i],q[i]=dp[ch[now][1]][i];
FMT_pre(p);FMT_pre(q);
FOR(i,0,lim-1) p[i]=1ll*p[i]*q[i]%mod;
IFMT_pre(p);
FOR(i,0,lim-1) dp[now][i]=(dp[now][i]+p[i])%mod;
}
if(val[now]!='&'){
FOR(i,0,lim-1) p[i]=dp[ch[now][0]][i],q[i]=dp[ch[now][1]][i];
FMT_suf(p);FMT_suf(q);
FOR(i,0,lim-1) p[i]=1ll*p[i]*q[i]%mod;
IFMT_suf(p);
FOR(i,0,lim-1) dp[now][i]=(dp[now][i]+p[i])%mod;
}
}
}
int main(){
scanf("%s",s+1);
n=strlen(s+1);
m=read();
FOR(i,0,m-1){
FOR(j,0,3) a[i]=2*a[i]+read();
b[i]=read();
}
FOR(i,1,n){
if(s[i]=='(') stk[++tp]=i;
else if(s[i]==')') with[i]=stk[tp],with[stk[tp]]=i,tp--;
}
FOR(i,1,n) if(s[i]!='(' && s[i]!=')'){
id[i]=++cnt;
val[cnt]=s[i];
int j=i;
while(j<=n && (s[j]!=')' || with[j]>i)) j++;
id[j]=id[with[j]]=cnt;
}
FOR(i,1,n) if(s[i]!='(' && s[i]!=')'){
if(s[i-1]==')' && s[i+1]=='('){
fa[id[i-1]]=fa[id[i+1]]=id[i];
ch[id[i]][0]=id[i-1];
ch[id[i]][1]=id[i+1];
op[id[i]]=1;
}
}
FOR(i,1,cnt) if(!fa[i]) assert(!rt),rt=i;
dfs(rt);
int ans=0;
FOR(i,0,lim-1){
bool flag=true;
FOR(j,0,m-1) if(((i>>a[j])&1)!=b[j]) flag=false;
if(flag) ans=(ans+dp[rt][i])%mod;
}
printf("%d\n",ans);
}
CF582E Boolean Function(DP,状态压缩,FMT)的更多相关文章
- HDU 4336 Card Collector (期望DP+状态压缩 或者 状态压缩+容斥)
题意:有N(1<=N<=20)张卡片,每包中含有这些卡片的概率,每包至多一张卡片,可能没有卡片.求需要买多少包才能拿到所以的N张卡片,求次数的期望. 析:期望DP,是很容易看出来的,然后由 ...
- HDU 1074 Doing Homework (dp+状态压缩)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1074 题目大意:学生要完成各科作业, 给出各科老师给出交作业的期限和学生完成该科所需时间, 如果逾期一 ...
- hdu_4352_XHXJ's LIS(数位DP+状态压缩)
题目连接:hdu_4352_XHXJ's LIS 题意:这题花大篇篇幅来介绍电子科大的一个传奇学姐,最后几句话才是题意,这题意思就是给你一个LL范围内的区间,问你在这个区间内最长递增子序列长度恰为K的 ...
- hdu 4352 数位dp + 状态压缩
XHXJ's LIS Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- 【bzoj1076】[SCOI2008]奖励关 期望dp+状态压缩dp
题目描述 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必须在抛出下一个宝物之前做出选择,且现在决定不吃的宝物以后也不能再 ...
- hdu4336 Card Collector(概率DP,状态压缩)
In your childhood, do you crazy for collecting the beautiful cards in the snacks? They said that, fo ...
- dp状态压缩
dp状态压缩 动态规划本来就很抽象,状态的设定和状态的转移都不好把握,而状态压缩的动态规划解决的就是那种状态很多,不容易用一般的方法表示的动态规划问题,这个就更加的难于把握了.难点在于以下几个方面:状 ...
- 洛谷 1052 dp 状态压缩
洛谷1052 dp 状态压缩 传送门 (https://www.luogu.org/problem/show?pid=1052#sub) 做完这道题之后,感觉涨了好多见识,以前做的好多状压题目都是将一 ...
- NOIP2005过河[DP 状态压缩]
题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数 ...
- ACM: HDU 5418 Victor and World - Floyd算法+dp状态压缩
HDU 5418 Victor and World Time Limit:2000MS Memory Limit:131072KB 64bit IO Format:%I64d & ...
随机推荐
- Dynamics 365 Customer Engagement V9.X新引入的自动编号属性介绍
我是微软Dynamics 365 & Power Platform方面的工程师罗勇,也是2015年7月到2018年6月连续三年Dynamics CRM/Business Solutions方面 ...
- macOS Catalina Kernel panic 因为意外而重新启动
0x00 What's Happend? 我的 MacBook Air 在升级到 Catalina 之后,经常在休眠模式重启,随后在桌面上显示"因为意外而重新启动"的信息,以下是跟 ...
- postgres centos 创建数据库 创建用户
一个小的流程关于如何创建数据库和用户,用以加强印象,以及留档备份 一.创建账户 1.登录postgres账户 su postgres 2.进入psql 指令 psql 3.创建用户 create US ...
- java中使用lambda表达式
使用lambda表达式能够使复杂的编写方式变的简单 lambda表达式的语法 (parameters) -> expression 或 (parameters) ->{ statement ...
- nginx基础(1)
目录 一.概念 基础概念 响应码 请求和响应报文的格式 http无连接 我叫张贺,贪财好色.一名合格的LINUX运维工程师,专注于LINUX的学习和研究,曾负责某中型企业的网站运维工作,爱好佛学和跑步 ...
- 【使用篇二】SpringBoot的日志体系及如何开启logback日志(15)
抄自:https://blog.csdn.net/liujun03/article/details/82684209 Java应用中,日志一般分为以下5个级别(从高到低): ERROR 错误信息 WA ...
- ModuleNotFoundError: No module named 'xxx'; 'xxx' is not a package
错误: ModuleNotFoundError: No module named 'xxx'; 'xxx' is not a package 通过pycharm对脚本进行debug时,出现了如下错: ...
- 一文掌握 Lambda 表达式
本文将介绍 Java 8 新增的 Lambda 表达式,包括 Lambda 表达式的常见用法以及方法引用的用法,并对 Lambda 表达式的原理进行分析,最后对 Lambda 表达式的优缺点进行一个总 ...
- 图解Numpy的tile函数
Numpy的tile(A, reps)函数,就是将原矩阵横向.纵向地复制.tile是瓷砖的意思,顾名思义,这个函数就是把数组像瓷砖一样铺展开来. 举个例子,原矩阵: 横向铺展: 纵向铺展: 横向铺 ...
- jQuery 源码解析(二十七) 样式操作模块 坐标详解
样式操作模块可用于管理DOM元素的样式.坐标和尺寸,本节讲解一下坐标这一块. 对于坐标来说,jQuery提供了一个offset方法用于获取第一个匹配元素的坐标或者设置所有匹配元素的坐标,还有offse ...