【洛古 P8815】[CSP-J 2022] 逻辑表达式
[CSP-J 2022] 逻辑表达式
题目描述
逻辑表达式是计算机科学中的重要概念和工具,包含逻辑值、逻辑运算、逻辑运算优先级等内容。
在一个逻辑表达式中,元素的值只有两种可能: 0 0 0(表示假)和 1 1 1(表示真)。元素之间有多种可能的逻辑运算,本题中只需考虑如下两种:“与”(符号为 &
)和“或”(符号为 |
)。其运算规则如下:
0 & 0 = 0 & 1 = 1 & 0 = 0 0 \mathbin{\&} 0 = 0 \mathbin{\&} 1 = 1 \mathbin{\&} 0 = 0 0&0=0&1=1&0=0, 1 & 1 = 1 1 \mathbin{\&} 1 = 1 1&1=1;
0 ∣ 0 = 0 0 \mathbin{|} 0 = 0 0∣0=0, 0 ∣ 1 = 1 ∣ 0 = 1 ∣ 1 = 1 0 \mathbin{|} 1 = 1 \mathbin{|} 0 = 1 \mathbin{|} 1 = 1 0∣1=1∣0=1∣1=1。
在一个逻辑表达式中还可能有括号。规定在运算时,括号内的部分先运算;两种运算并列时,&
运算优先于 |
运算;同种运算并列时,从左向右运算。
比如,表达式 0|1&0
的运算顺序等同于 0|(1&0)
;表达式 0&1&0|1
的运算顺序等同于 ((0&1)&0)|1
。
此外,在 C++ 等语言的有些编译器中,对逻辑表达式的计算会采用一种“短路”的策略:在形如 a&b
的逻辑表达式中,会先计算 a
部分的值,如果 a = 0 a = 0 a=0,那么整个逻辑表达式的值就一定为 0 0 0,故无需再计算 b
部分的值;同理,在形如 a|b
的逻辑表达式中,会先计算 a
部分的值,如果 a = 1 a = 1 a=1,那么整个逻辑表达式的值就一定为 1 1 1,无需再计算 b
部分的值。
现在给你一个逻辑表达式,你需要计算出它的值,并且统计出在计算过程中,两种类型的“短路”各出现了多少次。需要注意的是,如果某处“短路”包含在更外层被“短路”的部分内则不被统计,如表达式 1|(0&1)
中,尽管 0&1
是一处“短路”,但由于外层的 1|(0&1)
本身就是一处“短路”,无需再计算 0&1
部分的值,因此不应当把这里的 0&1
计入一处“短路”。
输入格式
输入共一行,一个非空字符串 s s s 表示待计算的逻辑表达式。
输出格式
输出共两行,第一行输出一个字符 0
或 1
,表示这个逻辑表达式的值;第二行输出两个非负整数,分别表示计算上述逻辑表达式的过程中,形如 a&b
和 a|b
的“短路”各出现了多少次。
样例 #1
样例输入 #1
0&(1|0)|(1|1|1&0)
样例输出 #1
1
1 2
样例 #2
样例输入 #2
(0|1&0|1|1|(1|1))&(0&1&(1|0)|0|1|0)&0
样例输出 #2
0
2 3
提示
【样例解释 #1】
该逻辑表达式的计算过程如下,每一行的注释表示上一行计算的过程:
0&(1|0)|(1|1|1&0)
=(0&(1|0))|((1|1)|(1&0)) //用括号标明计算顺序
=0|((1|1)|(1&0)) //先计算最左侧的 &,是一次形如 a&b 的“短路”
=0|(1|(1&0)) //再计算中间的 |,是一次形如 a|b 的“短路”
=0|1 //再计算中间的 |,是一次形如 a|b 的“短路”
=1
【样例 #3】
见附件中的 expr/expr3.in
与 expr/expr3.ans
。
【样例 #4】
见附件中的 expr/expr4.in
与 expr/expr4.ans
。
【数据范围】
设 ∣ s ∣ \lvert s \rvert ∣s∣ 为字符串 s s s 的长度。
对于所有数据, 1 ≤ ∣ s ∣ ≤ 10 6 1 \le \lvert s \rvert \le {10}^6 1≤∣s∣≤106。保证 s s s 中仅含有字符 0
、1
、&
、|
、(
、)
且是一个符合规范的逻辑表达式。保证输入字符串的开头、中间和结尾均无额外的空格。保证 s s s
中没有重复的括号嵌套(即没有形如 ((a))
形式的子串,其中 a
是符合规范的逻辑表
达式)。
测试点编号 | ∣ s ∣ ≤ \lvert s \rvert \le ∣s∣≤ | 特殊条件 |
---|---|---|
1 ∼ 2 1 \sim 2 1∼2 | 3 3 3 | 无 |
3 ∼ 4 3 \sim 4 3∼4 | 5 5 5 | 无 |
5 5 5 | 2000 2000 2000 | 1 |
6 6 6 | 2000 2000 2000 | 2 |
7 7 7 | 2000 2000 2000 | 3 |
8 ∼ 10 8 \sim 10 8∼10 | 2000 2000 2000 | 无 |
11 ∼ 12 11 \sim 12 11∼12 | 10 6 {10}^6 106 | 1 |
13 ∼ 14 13 \sim 14 13∼14 | 10 6 {10}^6 106 | 2 |
15 ∼ 17 15 \sim 17 15∼17 | 10 6 {10}^6 106 | 3 |
18 ∼ 20 18 \sim 20 18∼20 | 10 6 {10}^6 106 | 无 |
其中:
特殊性质 1 为:保证 s s s 中没有字符 &
。
特殊性质 2 为:保证 s s s 中没有字符 |
。
特殊性质 3 为:保证 s s s 中没有字符 (
和 )
。
【提示】
以下给出一个“符合规范的逻辑表达式”的形式化定义:
- 字符串
0
和1
是符合规范的; - 如果字符串
s
是符合规范的,且s
不是形如(t)
的字符串(其中t
是符合规范的),那么字符串(s)
也是符合规范的; - 如果字符串
a
和b
均是符合规范的,那么字符串a&b
、a|b
均是符合规范的; - 所有符合规范的逻辑表达式均可由以上方法生成。
解题思路
参考文章 表达式的转换我们可以把输入的逻辑表达式转换成后缀表达式。设l1,l2表示共发生l1个‘&’ 短路,l2个‘|’短路.
然后在计算的时候也开两个标记a,b,表示得到每个计算结果通过了a个‘&’ 短路,b个‘|’短路,
分类讨论
当前计算为 p 1 p1 p1& p 2 = p p2=p p2=p时:
- 若p1==0,发生‘&’ 短路,p2中的短路不算了,p继承p1中的短路, p . a = p 1. a + 1 , p . b = p 1. b p.a=p1.a+1,p.b=p1.b p.a=p1.a+1,p.b=p1.b
- 对于总短路l1,l2,要把p2中发生的短路减去 l 1 − = p 2. a , l 2 − = p 2. b ; l1-=p2.a,l2-=p2.b; l1−=p2.a,l2−=p2.b;
当前计算为 p 1 p1 p1| p 2 = p p2=p p2=p时:
- 若p1==1,发生‘|’ 短路,p2中的短路不算了,p继承p1中的短路, p . a = p 1. a + 1 , p . b = p 1. b p.a=p1.a+1,p.b=p1.b p.a=p1.a+1,p.b=p1.b
- 对于总短路l1,l2,要把p2中发生的短路减去 l 1 − = p 2. a , l 2 − = p 2. b ; l1-=p2.a,l2-=p2.b; l1−=p2.a,l2−=p2.b;
代码
//代码中的p1,p2与解题思路中的相反
#include<bits/stdc++.h>
#define ll long longusing namespace std;
char a[1100000];
int l,l1,l2;struct cc{int a,b;
};pair<int,cc>p1,p2,p;
stack<int>num;
stack<char>dat;
stack<char>op;
stack<pair<int,cc> >dat2;int check(char c){switch(c){case ')':return 0;case '(':return 0;case '|':return 1;case '&':return 2;default: return -1; }
}void js(char c){cc w;p1=dat2.top();dat2.pop();p2=dat2.top();dat2.pop();if(c=='&'){p.first=p1.first&p2.first;p.second.a=p.second.a;p.second.b=0; if(p2.first==0){l1++;p.second.a=p2.second.a+1;p.second.b=p2.second.b;l1-=p1.second.a,l2-=p1.second.b; }else{p.second.a=p1.second.a+p2.second.a;p.second.b=p1.second.b+p2.second.b;}} if(c=='|'){p.first=p1.first|p2.first; if(p2.first==1){l2++;p.second.a=p2.second.a;p.second.b=p2.second.b+1; l1-=p1.second.a,l2-=p1.second.b;}else{p.second.a=p1.second.a+p2.second.a;p.second.b=p1.second.b+p2.second.b;}}
}
void work2(){while(!dat.empty()){char t=dat.top();dat.pop();op.push(t);}while(!op.empty()){char c=op.top(); if(c>='0'&&c<='9'){cc x;x.a=0,x.b=0;dat2.push(make_pair(c-'0',x));op.pop();}else {js(c);dat2.push(p); op.pop();}}printf("%d\n",dat2.top());printf("%d %d",l1,l2);
}void work1(){for(int i=1;i<=l;i++){if(a[i]>='0'&&a[i]<='9')dat.push(a[i]);else {if(op.empty()){op.push(a[i]);continue;}char c=op.top();if(a[i]=='('){ op.push(a[i]);continue;}if(a[i]==')'){while(c!='('&&!op.empty()){dat.push(c);op.pop(); if(!op.empty()) c=op.top();}op.pop();continue;}if(check(a[i])>check(c)){op.push(a[i]);}else{while(check(a[i])<=check(c)&&!op.empty()){dat.push(c);op.pop(); if(!op.empty()) c=op.top();}op.push(a[i]);}}}while(!op.empty()){dat.push(op.top()); op.pop();}
}int main(){scanf("%s",a+1);l=strlen(a+1);work1();work2();
}
【洛古 P8815】[CSP-J 2022] 逻辑表达式相关推荐
- 洛古最简单50题解(1-10)
做为一名新手,首先要过一过题,找找成就感.(大佬略过).下面附上洛古最简单50题(大佬略过).以及最麻烦 AC代码,至少AC了. 目录·列表: 洛古最简单50题解(1-10) 洛古最简单50题解(11 ...
- 洛古最简单50题解(41-50)
做为一名新手,首先要过一过题,找找成就感.(大佬略过).下面附上洛古最简单50题(大佬略过).以及最麻烦 AC代码,至少AC了. 目录·列表: 洛古最简单50题解(1-10) 洛古最简单50题解(11 ...
- 洛古最简单50题解(21-30)
做为一名新手,首先要过一过题,找找成就感.(大佬略过).下面附上洛古最简单50题(大佬略过).以及最麻烦 AC代码,至少AC了. 目录·列表: 洛古最简单50题解(1-10) 洛古最简单50题解(11 ...
- 洛古最简单50题解(31-40)
做为一名新手,首先要过一过题,找找成就感.(大佬略过).下面附上洛古最简单50题(大佬略过).以及最麻烦 AC代码,至少AC了. 目录·列表: 洛古最简单50题解(1-10) 洛古最简单50题解(11 ...
- 洛古最简单50题解(11-20)
做为一名新手,首先要过一过题,找找成就感.(大佬略过).下面附上洛古最简单50题(大佬略过).以及最麻烦 AC代码,至少AC了. 目录·列表: 洛古最简单50题解(1-10) 洛古最简单50题解(11 ...
- CSP - J 2020 T1 优秀的拆分
https://www.luogu.com.cn/problem/P7071 /* CSP - J 2020 T1 优秀的拆分 https://www.luogu.com.cn/problem/P70 ...
- 洛古 P1373 小a和uim之大逃离
P1373 小a和uim之大逃离 题目提供者lzn 标签 动态规划 洛谷原创 难度 提高+/省选- 题目背景 小a和uim来到雨林中探险.突然一阵北风吹来,一片乌云从北部天边急涌过来,还伴着一道道闪电 ...
- 2020 CSP - J初赛 题解
目录 写在前面的话 题面 题解 答案合集 单项选择题 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 阅读程序题 一 16 17 18 19 20 21 二 22 23 24 ...
- 洛古P2392-kkksc03考前临时抱佛脚
背包问题 1.kkksc03考前临时抱佛脚 题目 题目背景 kkksc03 的大学生活非常的颓废,平时根本不学习.但是,临近期末考试,他必须要开始抱佛脚,以求不挂科. 题目描述 这次期末考试,kkks ...
最新文章
- ESX 4 不能使用SSH登录的解决
- 应用程序框架实战三十六:CRUD实战演练介绍
- DevExpress v17.2新版亮点—WinForms篇(四)
- C++Primer学习笔记:第8章 IO库
- Web开发如何实现Tomcat等服务器热部署不用重启
- SIP中第三方呼叫控制(3PCC)建立流程
- WebSocket connection to,Error during WebSocket handshake: Unexpected response code: 404
- oracle PS/SQL DEVELOPER
- 华为手机鸿蒙系统下载,华为鸿蒙系统手机版
- 『水晶报表』实现打印
- 数控系统数据采集协同架构,集成马扎克(mazak)、西门子(Siemens)、海德汉(heidenhain)、广数、凯恩帝(knd)、三菱、海德汉、兄弟、哈斯、宝元、新代、发那科(Fanuc)、华中
- 反编译工具apktool
- 0CTF 2016 RSA?(未完成)
- 金蝶云苍穹笔记(一)
- ubuntu 下解压带密码的rar压缩包
- 《大话移动通信(第2版)》赠书活动名单公告
- cad转dxf格式文件太大_高版本的CAD图纸怎么转换成低版本的DXF格式?
- 微软一个数据中心有多少服务器,微软晒Windows数据中心级机器 一共拥有多达896个物理核心和1792个逻辑核心...
- mmse评估量表_简易精神状态评价量表(MMSE量表)
- week4 day3/4 常用模块
热门文章
- 2023 红明谷杯 --- Crypto It Takes Two! wp
- MyEclipse的使用小技巧,您get到了吗?
- 正则表达式-支付宝账号验证
- 450g吐司烘烤温度_450g吐司要烤多少时间?
- 一篇文章讲清楚VO,BO,PO,DO,DTO的区别
- 显示器屏幕有时黑几秒又恢复相关解决方法
- Android系统的JNI原理分析(二)- 数据类型转换和方法签名
- css3宽度变大动画_动画演示14种流量计的工作原理,真涨见识!
- 百度API实现logo商标识别接口介绍
- html文字冒险游戏,文字类谍战交互游戏能有这么好玩?全靠这部超神原作