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 & ...
随机推荐
- Geodesic 什么是“测地线的”?
确定是使用上椭球体(测地线)还是平地上(平面)的最短路径.强烈建议将 Geodesic 方法用于在不适合进行距离测量的坐标系(例如 Web 墨卡托或任何地理坐标系)中存储的数据,以及任何地理区域跨度较 ...
- Jerome: Vulnhub Walkthrough
nmap 扫描探测: ╰─ nmap -p1-65535 -sV -A -O -sT 10.10.202.135Starting Nmap 7.70 ( https://nmap.org ) at 2 ...
- 一文解读Redis (转)
本文由葡萄城技术团队编撰并首发 转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 引言 在Web应用发展的初期,那时关系型数据库受到了较为广泛的关注和应用,原 ...
- 表单生成器(Form Builder)之mongodb表单数据查询——返回分页数据和总条数
上一篇笔记将开始定义的存储结构处理了一下,将FormItems数组中的表单项都拿到mongodb document的最外层,和以前的关系型数据类似,之不过好多列都是动态的,不固定,不过这并没有什么影响 ...
- LeetCode刷题--两数相加(中等)
题目描述 给出两个 非空 的链表用来表示两个非负的整数.其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字. 如果,我们将这两个数相加起来,则会返回一个新的链表来表 ...
- 基于django的个人博客网站建立(五)
基于django的个人博客网站建立(五) 前言 网站效果可点击这里访问 之前鸽了两天,今天继续再写点 主要内容 今天加了个展示照片的功能,就叫他生活记录吧 先建表 class Record(model ...
- java之匿名内部类
Person.java package insof; public class Person extends Object{ String name; static int age; public P ...
- 利用Python进行数据分析-Pandas(第六部分-数据聚合与分组运算)
对数据集进行分组并对各组应用一个函数(无论是聚合还是转换),通常是数据分析工作中的重要环节.在将数据集加载.融合.准备好之后,通常是计算分组统计或生成透视表.pandas提供了一个灵活高效的group ...
- NetCore WebAPI开发探索
一.创建项目 新建api项目: 建好之后,一个默认的控制器已经有了: 运行就可以直接访问get接口获取数据: 二.跨平台部署 部署方面,微软已经完善的很好了,基本上算是傻瓜式操作.项目右键选择发布 ...
- ORM和Mybatis
ORM框架 概述 在学习MyBatis之前,先来看看什么是ORM框架. ORM全称Object/Relation Mapping,对象/关系数据库映射,功能为完成对象的编程语言到关系数据库的映射,可以 ...