P7324 [WC2021] 表达式求值

闲话

WC2021 我只得了 20 分,三道题总共 20 分。我是下场了突然后知后觉这件事的,主要原因是我开了 C++11,然后 T1 T2 都没分了。在洛谷上测本来能拿银牌的。T1 的乱搞能拿 48,还挺高的。

幸亏咱们陕西省选不看冬令营成绩。幸亏是在省选前犯的这个错误。告诫后人和自己,写题前一定要看编译选项,否则只能后悔莫及。

T2 场上写的是不带问号的 \(O(n|E|)\) 和带问号 \(O(n|E|m^2)\) 的 70 分暴力。后者可以用 minmax 卷积优化掉一个 \(m\) 获得 85 分的好成绩。

思路

首先要会上面提到的带问号的暴力,不需要优化。

我在场上写的是这样子的:

int a[maxn][10],but[maxn][10],tmp[10];
int* solve(int l,const int &r,const int *a){
l++;
int *ans=s[l]=='('?solve(l,to[l],a):but[l];//这里的 to 是预处理的括号匹配位置,完全可以用把 l 设成全局变量的方法解决
l=s[l]=='('?to[l]+1:l+1;
while(l+1<r){
char op=s[l++];
int *res=s[l]=='('?solve(l,to[l],a):but[l];
l=s[l]=='('?to[l]+1:l+1;
switch(op){
case '<':{
memset(tmp,0,sizeof tmp);
for(int i=0;i<m;i++) for(int j=0;j<m;j++) if(a[i]<a[j]) tmp[i]=(tmp[i]+1ll*ans[i]*res[j])%mod;
else tmp[j]=(tmp[j]+1ll*ans[i]*res[j])%mod;
for(int i=0;i<m;i++) ans[i]=tmp[i];
break;
}
case '>':{
memset(tmp,0,sizeof tmp);
for(int i=0;i<m;i++) for(int j=0;j<m;j++) if(a[i]>a[j]) tmp[i]=(tmp[i]+1ll*ans[i]*res[j])%mod;
else tmp[j]=(tmp[j]+1ll*ans[i]*res[j])%mod;
for(int i=0;i<m;i++) ans[i]=tmp[i];
break;
}
case '?':{
memset(tmp,0,sizeof tmp);
for(int i=0;i<m;i++) for(int j=0;j<m;j++) tmp[i]=(tmp[i]+1ll*ans[i]*res[j])%mod,tmp[j]=(tmp[j]+1ll*ans[i]*res[j])%mod;
for(int i=0;i<m;i++) ans[i]=tmp[i];
break;
}
}
}
return ans;
}
inline void work(){
for(int i=0;i<m;i++) for(int j=1;j<=n;j++) a[j][i]=star::a[i][j-1];
for(int i=1;i<=n;i++){
for(int j=1;j<=len;j++) for(int k=0;k<m;k++) but[j][k]=(isdigit(s[j]) and s[j]-'0'==k);
int* res=solve(1,len,a[i]);
for(int j=0;j<m;j++)
ans=(ans+1ll*res[j]*a[i][j])%mod;
}
printf("%d\n",ans);
}

即把 \(n\) 组数分开来讨论,求每种数贡献的方案数,最后乘上值加起来。

那么问题就在于求每种数的方案数,这样的工作我们需要做 \(n\) 遍,非常不划算。

然后发现,对于大小关系相同的每组数,每种数贡献的方案数是相同的,改变的只是它们的值,我们没有必要每次都算一遍方案数。

发现,一个数在一组数中的相对大小关系的情况总数只有 \(2^m\) 种,即是否比该数大。

那么我们预处理出一个数的所有相对大小关系总数(这个部分是 \(O(2^m|E|)\) 的),然后对于 \(n\) 组数每个枚举 \(m\) 个数差分统计答案即可(差分得到该数的方案数。此部分是 \(O(nmlogm)\) 的)。

代码

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cctype>
#include<cstring>
#include<cmath>
#include<utility>
using namespace std;
inline int read(){
int w=0,x=0;char c=getchar();
while(!isdigit(c))w|=c=='-',c=getchar();
while(isdigit(c))x=x*10+(c^48),c=getchar();
return w?-x:x;
}
namespace star
{
const int maxn=5e4+10,maxm=12,S=1050,mod=1e9+7;
int n,m,a[maxn][maxm],D,f[S],pos,b[maxm],ans;
char s[maxn];
pair<int,int> solve(){
int num=0;bool ok=true;char lst='>';
pair<int,int> ans(1,0),now,tmp;
while(lst!=')' and ++pos)
if(isdigit(s[pos])) num=s[pos]^48;
else if(s[pos]=='(') now=solve(),ok=false;
else ok and (now=make_pair((D>>num)&1,(D>>num)&1^1),0),
ok=true,tmp=make_pair(0,0),
lst!='>' and (tmp.second=(tmp.second+1ll*ans.second*now.second)%mod,tmp.first=(tmp.first+1ll*(ans.first+ans.second)*(now.first+now.second)-1ll*ans.second*now.second%mod+mod)%mod),
lst!='<' and (tmp.first=(tmp.first+1ll*ans.first*now.first)%mod, tmp.second=(tmp.second+1ll*(ans.first+ans.second)*(now.first+now.second)-1ll*ans.first*now.first%mod+mod)%mod),
ans=tmp,lst=s[pos],num=0;
return ans;
}
inline bool cmp(const int &x,const int &y){return a[pos][x]<a[pos][y];}
inline void work(){
n=read(),m=read();
for(int i=0;i<m;i++) for(int j=0;j<n;j++) a[j][i]=read();
scanf("%s",s+1);s[strlen(s+1)+1]=')';
for(D=0;D<(1<<m);D++) pos=0,f[D]=solve().second;
for(int i=0;i<n;i++){
for(int j=0;j<m;j++) b[j]=j;
pos=i;sort(b,b+m,cmp);
for(int j=0,d=0;j<m;d|=(1<<b[j++]))
ans=(ans+1ll*(a[i][b[j]]-(j?a[i][b[j-1]]:0))*f[d])%mod;
}
printf("%d\n",ans);
}
}
signed main(){
star::work();
return 0;
}

P7324 [WC2021] 表达式求值的更多相关文章

  1. 洛谷 P7324 - [WC2021] 表达式求值(状压+dp)

    题面传送门 现场人傻系列-- 首先建出 \(E\) 的表达式树,具体来说表达式的每一个叶子节点表示一个数组 \(A_i\),每一个非叶子节点都表示一次运算,它的值表示左右儿子进行该运算后得到的结果.这 ...

  2. [WC2021] 表达式求值

    考虑我们显然可以对每位分开求解,考虑求出最终答案是\(A_i\)的方案数. 那么我们发现我们这样\(dp\)的话,显然不太行! 会有一个\(i\)的复杂度 但是如果我们做大于等于的话,就只用一遍\(d ...

  3. 表达式求值(noip2015等价表达式)

    题目大意 给一个含字母a的表达式,求n个选项中表达式跟一开始那个等价的有哪些 做法 模拟一个多项式显然难以实现那么我们高兴的找一些素数代入表达式,再随便找一个素数做模表达式求值优先级表 - ( ) + ...

  4. 用Python3实现表达式求值

    一.题目描述 请用 python3 编写一个计算器的控制台程序,支持加减乘除.乘方.括号.小数点,运算符优先级为括号>乘方>乘除>加减,同级别运算按照从左向右的顺序计算. 二.输入描 ...

  5. 数据结构算法C语言实现(八)--- 3.2栈的应用举例:迷宫求解与表达式求值

    一.简介 迷宫求解:类似图的DFS.具体的算法思路可以参考书上的50.51页,不过书上只说了粗略的算法,实现起来还是有很多细节需要注意.大多数只是给了个抽象的名字,甚至参数类型,返回值也没说的很清楚, ...

  6. nyoj305_表达式求值

    表达式求值 时间限制:3000 ms  |  内存限制:65535 KB 难度:3   描述 Dr.Kong设计的机器人卡多掌握了加减法运算以后,最近又学会了一些简单的函数求值,比如,它知道函数min ...

  7. 利用栈实现算术表达式求值(Java语言描述)

    利用栈实现算术表达式求值(Java语言描述) 算术表达式求值是栈的典型应用,自己写栈,实现Java栈算术表达式求值,涉及栈,编译原理方面的知识.声明:部分代码参考自茫茫大海的专栏. 链栈的实现: pa ...

  8. 数据结构--栈的应用(表达式求值 nyoj 35)

    题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=35 题目: 表达式求值 时间限制:3000 ms | 内存限制:65535 KB描述 AC ...

  9. NOIP2013普及组 T2 表达式求值

    OJ地址:洛谷P1981 CODEVS 3292 正常写法是用栈 #include<iostream> #include<algorithm> #include<cmat ...

随机推荐

  1. HashMap底层实现原理及面试常见问题

    HashMap底层源码分析 1.HashMap底层采用的存储结构 1.在JDK1.7及之前采用的存储结构是数组+链表 2.到了JDK1.8之后采用的是数组+链表+红黑树 2.HashMap实现的原理 ...

  2. CENTOS7 安装 SYNCTHING

    本地电脑需要同步远程数据,安装syncthing 测试 1:下载 wget https://github.com/syncthing/syncthing/releases/download/v1.5. ...

  3. 5000字长文,kurryluo 的自学编程之路

    我是程序员.大众口中非科班的那种,带着高中时期对二进制的恐惧,在大学参加科研比赛后保研,再到和校友一起创业,现在在某大型互联网公司做前端开发,一路走来都是靠自己学习. 前端框架 VUE 的作者尤大说过 ...

  4. Mysql优化(出自官方文档) - 第二篇

    Mysql优化(出自官方文档) - 第二篇 目录 Mysql优化(出自官方文档) - 第二篇 1 关于Nested Loop Join的相关知识 1.1 相关概念和算法 1.2 一些优化 1 关于Ne ...

  5. LCD1602液晶显示模块的单片机驱动深入详解之硬件篇

    (本文以HD44780主控芯片的LCD1602为蓝本进行描述,其中的截图也来自HD44780数据手册,用户可自行搜索其datasheet,有部分整理网上的,但绝对要比你看到的要深入得多) 一.接口 L ...

  6. C++中指针与引用详解

    在计算机存储数据时必须要知道三个基本要素:信息存储在何处?存储的值为多少?存储的值是什么类型?因此指针是表示信息在内存中存储地址的一类特殊变量,指针和其所指向的变量就像是一个硬币的两面.指针一直都是学 ...

  7. Kubernetes ConfigMap详解,多种方式创建、多种方式使用

    我最新最全的文章都在南瓜慢说 www.pkslow.com,欢迎大家来喝茶! 1 简介 配置是程序绕不开的话题,在Kubernetes中使用ConfigMap来配置,它本质其实就是键值对.本文讲解如何 ...

  8. TS基础应用 & Hook中的TS

    说在前面 本文难度偏中下,涉及到的点大多为如何在项目中合理应用ts,小部分会涉及一些原理,受众面较广,有无TS基础均可放心食用. **>>>> 阅完本文,您可能会收获到< ...

  9. Redis的主从数据一致性

    我们学习了 AOF 和 RDB,如果 Redis 发生了宕机,它们可以分别通过回放日志和重新读入 RDB 文件的方式恢复数据,从而保证尽量少丢失数据,提升可靠性.不过,即使用了这两种方法,也依然存在服 ...

  10. 13、linux中用户和用户组

    linux是多用户多进程的系统: 每个文件和进程都需要应对一个用户和用户组: linux系统通过uid和gid来识别用户和组的: 一个用户必须要有唯一的uid和一个主组来识别身份,不同的用户可以使用同 ...