【题目链接】

https://www.luogu.org/problem/P1310

题目描述

对于1 位二进制变量定义两种运算:

运算的优先级是:

  1. 先计算括号内的,再计算括号外的。

  2. “× ”运算优先于“⊕”运算,即计算表达式时,先计算× 运算,再计算⊕运算。例如:计算表达式A⊕B × C时,先计算 B × C,其结果再与 A 做⊕运算。

现给定一个未完成的表达式,例如_+(_*_),请你在横线处填入数字00或者11 ,请问有多少种填法可以使得表达式的值为00。

输入格式

共 2 行。

第1 行为一个整数 LL,表示给定的表达式中除去横线外的运算符和括号的个数。

第2 行为一个字符串包含 LL 个字符,其中只包含’(’、’)’、’+’、’*’这44 种字符,其中’(’、’)’是左右括号,’+’、’*’分别表示前面定义的运算符“⊕”和“×”。这行字符按顺序给出了给定表达式中除去变量外的运算符和括号。

输出格式

共1 行。包含一个整数,即所有的方案数。注意:这个数可能会很大,请输出方案数对1000710007取模后的结果。

输入输出样例

输入 #1

4
+(*)
输出 #1

说明/提示

【输入输出样例说明】

给定的表达式包括横线字符之后为:_+(_*_)

在横线位置填入(0 、0 、0) 、(0 、1 、0) 、(0 、0 、1) 时,表达式的值均为0 ,所以共有3种填法。

【题解】

1、转变为后缀表达式的形式,然后,如果是操作数要添加一个数字,记住第一个位置要多添加一个‘.’

2、然后进行推导,把四种情况的转移状态写清楚。

参考题解中duyi

第一步:中缀转后缀

后缀表达式是什么呢?参见洛谷P1449

这里先给大家介绍一下中缀表达式转后缀表达式的一般方法:

中缀表达式a + b*c + (d * e + f) * ga+b∗c+(d∗e+f)∗g,其转换成后缀表达式则为a b c * + d e * f + g * +abc∗+de∗f+g∗+。

转换过程需要用到栈,具体过程如下:

1)如果遇到操作数,我们就直接将其输出。

2)如果遇到操作符,则我们将其放入到栈中,遇到左括号时我们也将其放入栈中。

3)如果遇到一个右括号,则将栈元素弹出,将弹出的操作符输出直到遇到左括号为止。注意,左括号只弹出并不输出。

4)如果遇到任何其他的操作符,如(“+”, “*”,“(”)等,从栈中弹出元素直到遇到发现更低优先级的元素(或者栈为空)为止。弹出完这些元素后,才将遇到的操作符压入到栈中。有一点需要注意,只有在遇到" ) "的情况下我们才弹出" ( ",其他情况我们都不会弹出" ( "。

5)如果我们读到了输入的末尾,则将栈中所有元素依次弹出。

备注:本题中我们用一个"."来代表数字。扫描整个表达式(读入的字符串),如果当前位置不是括号(既不是左括号也不是右括号),就在后缀表达式里填一个"."表示这里应有一个数字。

第二步:DP

状态转移方程很好想:

如果当前是"*":

f[0][now]=f[0][now]∗f[0][last]+f[0][now]∗f[1][last]+f[1][now]∗f[0][last]

f[1][now]=f[1][now]∗f[1][now]

(注意两式顺序不能颠倒,因为1式中需要用到f[1][now]f[1][now]的原始值)

如果当前是"+":

f[1][now]=f[1][now]∗f[1][last]+f[0][now]∗f[1][last]+f[1][now]∗f[0][last]

f[0][now]=f[0][now]∗f[1][last]

(两式的顺序同样不能颠倒)

其中,f[i][j]表示j号数字是i的情况有多少种,初始化为1


【代码】

 #include<bits/stdc++.h>
using namespace std;
const int mod = ;
const int N = 1e5+; int f[][N] ; int priority( char s ){
switch ( s ){
case '+' : return ;
case '*' : return ;
case '(' :
case ')' : return -;
}
} void toSuffixString( string &s ){
int len = s.length(); string res = "." ;
stack < int > Op ; for(int i=;i<len ;i++){
/*if( s[i] == '.') {
res += ".";
}else */
if( s[i] == '(' || s[i] == '*' ){
Op.push(s[i]);
}else if( s[i] == ')' ){
while( !Op.empty() && Op.top() != '(' ){
res += Op.top();
Op.pop();
}
//出来的时候,栈顶为(.
Op.pop();
}else{
while( !Op.empty() && priority(Op.top()) >= priority(s[i]) ){
res += Op.top();
Op.pop();
}
//出来的时候,栈里面的运算符优先级低于当前位置.
Op.push( s[i] ) ;
}
if( !(s[i] == '(' || s[i] == ')') ){
res += ".";
}
}
while( !Op.empty() ){
res += Op.top();
Op.pop();
}
//cout << res << endl ;
s = res ;
} void Calc( string s ){
//cout << s << endl ;
int j = ;
int len = s.length();
for(int i= ; i < len ; i++ ){
if( s[i] == '.' ){
j ++ ;
f[][j] = f[][j] = ;
}else if( s[i] == '*' ){
j -- ;
f[][j] =(f[][j] * f[][j+] +
f[][j] * f[][j+] +
f[][j] * f[][j+] ) % mod ; f[][j] = (f[][j] * f[][j+]) % mod ; }else if( s[i] == '+' ){
j -- ;
f[][j] =(f[][j] * f[][j+] +
f[][j] * f[][j+] +
f[][j] * f[][j+] ) % mod ; f[][j] =(f[][j] * f[][j+] ) %mod ;
}
}
cout << f[][] << endl;
} int main(){
ios_base :: sync_with_stdio( false );
cin.tie(NULL) , cout.tie(NULL) ;
int n ;
string str ;
cin >> n >> str ;
toSuffixString(str);
Calc(str);
return ;
}

【数据结构】P1310 表达式的值的更多相关文章

  1. 洛谷 P1310 表达式的值 解题报告

    P1310 表达式的值 题目描述 对于1 位二进制变量定义两种运算: 运算的优先级是: 先计算括号内的,再计算括号外的. "× "运算优先于"⊕"运算,即计算表 ...

  2. 洛谷P1310 表达式的值

    P1310 表达式的值 题目描述 对于1 位二进制变量定义两种运算: 运算的优先级是: 先计算括号内的,再计算括号外的. “× ”运算优先于“⊕”运算,即计算表达式时,先计算× 运算,再计算⊕运算.例 ...

  3. P1310 表达式的值

    P1310 表达式的值 题解 1.假设有两个布尔变量 x , y  x0表示使得x=0的方案数 x1表示使得x=1的方案数 y0表示使得y=0的方案数 y1表示使得y=1的方案数 | 按位或 & ...

  4. 2019.06.17课件:[洛谷P1310]表达式的值 题解

    P1310 表达式的值 题目描述 给你一个带括号的布尔表达式,其中+表示或操作|,*表示与操作&,先算*再算+.但是待操作的数字(布尔值)不输入. 求能使最终整个式子的值为0的方案数. 题外话 ...

  5. luogu P1310 表达式的值

    题目描述 对于1 位二进制变量定义两种运算: 运算的优先级是: 先计算括号内的,再计算括号外的. “× ”运算优先于“⊕”运算,即计算表达式时,先计算× 运算,再计算⊕运算.例如:计算表达式A⊕B × ...

  6. 【洛谷P1310 表达式的值】

    题目链接 题目描述 对于1 位二进制变量定义两种运算: 运算的优先级是: 先计算括号内的,再计算括号外的. “× ”运算优先于“⊕”运算,即计算表达式时,先计算× 运算,再计算⊕运算.例如:计算表达式 ...

  7. 洛谷P1310 表达式的值 题解 栈/后缀表达式的应用

    题目链接:https://www.luogu.org/problem/P1310 本题涉及算法:栈.前缀表达式转后缀表达式,动态规划思想. 这道题目我思考了好长时间,第一时间让我做的话我也做不出来. ...

  8. 洛谷P1310 表达式的值——题解

    题目传送 题的难点:1.有运算优先级,不好判断.2.有破坏整体和谐性的讨厌的括号.3.不知道哪里要填数.4.要求方案数很大,搜索不会做呐. 发现难点1和2都是中缀表达式的缺点.转成后缀表达式后难点1. ...

  9. P1981 表达式求值

    P1981 表达式求值 题解 这个题联想一下  P1310 表达式的值  思路就是输入中缀式,转成后缀式,然后按后缀式计算,完美!!       but!! 会严重RE,因为你可能会输入中缀式的时候输 ...

随机推荐

  1. eclipse java

    1Java:Java是由Sun Microsystems公司推出的Java面向对象程序设计语言和Java平台的总称. 2.Eclipse:Eclipse 是一个开放源代码的.基于Java的可扩展开发平 ...

  2. 一个很好的开源视频播放器GiraffePlayer2(支持rtmp,rtsp,http,https)

    一个很好的开源视频播放器GiraffePlayer2(支持rtmp,rtsp,http,https) https://github.com/tcking/GiraffePlayer2 GiraffeP ...

  3. uni-app 使用Vuex+ (强制)登录

    一.在项目的根目录下新建一个store文件夹,然后在文件夹下新建一个index.js文件 二.在新建的index.js下引入vue和vuex,具体如下: //引入vue和vuex import Vue ...

  4. centos6.x一直停留在进度条的问题

    由于屏幕关闭加上长时间没有操作可能导致centos进行深度休眠,此时只能通过电源键来强制重启.可以通过修改X11的配置来禁止休眠,配置文件为/etc/X11/xorg.conf,注意修改此文件前记得备 ...

  5. iOS-SVPullToRefresh​下拉刷新,上拉加载(转)

    https://github.com/Sephiroth87/ODRefreshControl 类似刷新控件,类似qq动画的那种刷新. 一.下载第三方库 https://github.com/samv ...

  6. OPC Utgard的数据访问方式

    1.同步读取某个点位的值 Item项的read()方法 Server server = new Server(BaseConfiguration.getCLSIDConnectionInfomatio ...

  7. SpringBoot整合Redis及Redis工具类

    前言 想做一个秒杀项目,问了几个大佬要了项目视频,结果,自己本地实践的时候,发现不太一样,所以写下这篇,为以后做准备. 环境配置 IDE:IDEA 环境:Windows 数据库:Redis Maven ...

  8. SpringBoot:SpringBoot整合JdbcTemplate

    个人其实偏向于使用类似于JdbcTemplate这种的框架,返回数据也习惯于接受Map/List形式,而不是转化成对象,一是前后台分离转成json方便,另外是返回数据格式,数据字段可以通过SQL控制, ...

  9. Javascript 数组转无限级分类

    递归 var arr = [ {"id":1,"parent_id":0,"name":"Foods"}, {" ...

  10. 0《STL源码剖析》简介

    STL源码剖析 ----侯捷 STL主要包括六个组件: 1.配置器:负责空间配置和管理. 2.迭代器:扮演容器和算法之前的胶合剂,所谓“泛型指针”. 3.容器:各种数据结构,如vector,list, ...