NC50999 表达式计算4
NC50999 表达式计算4
题目
题目描述
给出一个表达式,其中运算符仅包含+,-,*,/,^(加 减 乘 整除 乘方)要求求出表达式的最终值
数据可能会出现括号情况,还有可能出现多余括号情况
数据保证不会出现 \(\geq 2^{31}\) 的答案
数据可能会出现负数情况
输入描述
仅一行,即为表达式
输出描述
仅一行,既为表达式算出的结果
示例1
输入
(2+2)^(1+1)
输出
16
备注
表达式总长度 \(\leq 30\)
题解
思路
知识点:分治。
按优先级给符号分类,记录每类符号不在括号内的最后一个出现的地方位置,随后找到优先级最低且符号位置存在的位置划分表达式为左右两边,对他们进行相同的操作,然后根据返回表达式值以及对应符号处理得到结果再返回到上一层。
关于匹配括号,引入括号计数器 \(cnt\) 如果为 \(0\) 则说明当前不在任何括号内,只有这时才能够确定符号位置。要注意可能会有多余括号问题,正常情况括号是互相匹配的,而多余的括号会使计数不为 \(0\) 。
在过程中如果遇到 \(cnt < 0\) ,说明中间有多余右括号,这种括号可以直接删除,不影响后面结果,因此这种情况下也可以确定符号位置,当然为了防止 \()(\) 的反向匹配导致优先级混乱,所以每次确定符号都要把 \(cnt\) 归零。如 $(1+3)) * (3+1) $ 的答案是 \(16\) 不归零就会得到 \(13\) 。
在遍历完算式后发现没有符号位置,则说明有最外层括号需要清除或者这个是个纯数字,进一步分类如果 \(cnt = 0\) ,再检查外侧是否有括号,决定是纯数字转化或者对称外层括号清除;如果 \(cnt > 0\) ,则有多余外层左括号清除; \(cnt<0\) ,则有多余外层右括号清除。
如存在位置,从低优先级检查位置,进行运算处理。
时间复杂度 平均:\(O(nlogn)\) 最差: \(O(n^2)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
using namespace std;
string s;
int StoI(int l, int r) {
int ans = 0;
for (int i = l;i <= r;i++) ans = ans * 10 + s[i] - '0';
return ans;
}
int calc(int l, int r) {
int cnt = 0;///记录括号数量
int pos[3] = { -1,-1,-1 };///分别对应+-,*/,^
for (int i = l;i <= r;i++) {///记录当前子算式不包括在括号内的符号
//())) + ()
if (s[i] == '(') cnt++;
else if (s[i] == ')')cnt--;
else if (cnt <= 0) {///判断符号是否在括号内,<0异常右端括号失配可以直接忽略(记得清空cnt)
cnt = 0;///取消右失配括号对后面的影响,原题数据不行
if (s[i] == '+' || s[i] == '-') pos[0] = i;
else if (s[i] == '*' || s[i] == '/') pos[1] = i;
else if (s[i] == '^') pos[2] = i;
}
}
// ((()+())
// (()+()
if (!~pos[0] && !~pos[1] && !~pos[2]) {///异常括号或者无符号纯数字导致的无括号外符号的处理
if (!cnt) {
if (s[l] == '(') return calc(l + 1, r - 1);///具有对称最外部括号
else return StoI(l, r);///没有括号的纯数字
}
else if (cnt > 0) return calc(l + 1, r);///去掉左括号
else if (cnt < 0) return calc(l, r - 1);///去掉右括号
}
else if (~pos[0]) {
if (s[pos[0]] == '+') return calc(l, pos[0] - 1) + calc(pos[0] + 1, r);
else if (s[pos[0]] == '-') return calc(l, pos[0] - 1) - calc(pos[0] + 1, r);
}
else if (~pos[1]) {
if (s[pos[1]] == '*') return calc(l, pos[1] - 1) * calc(pos[1] + 1, r);
else if (s[pos[1]] == '/') return calc(l, pos[1] - 1) / calc(pos[1] + 1, r);
}
else if (~pos[2]) return pow(calc(l, pos[2] - 1), calc(pos[2] + 1, r));
return 0;///底部返回值,不然过不了编译
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
cin >> s;
cout << calc(0, s.length() - 1) << '\n';
return 0;
}
NC50999 表达式计算4的更多相关文章
- .NET平台开源项目速览(8)Expression Evaluator表达式计算组件使用
在文章:这些.NET开源项目你知道吗?让.NET开源来得更加猛烈些吧!(第二辑)中,给大家初步介绍了一下Expression Evaluator验证组件.那里只是概述了一下,并没有对其使用和强大功能做 ...
- C# - 二叉树表达式计算
很早以前就写过双栈的表达式计算. 这次因为想深入学一下二叉树,网上都是些老掉牙的关于二叉树的基本操作. 感觉如果就学那些概念,没意思也不好记忆.于是动手写了一个表达式计算的应用例子. 这样学习印象才深 ...
- C#动态表达式计算
C#动态表达式计算 应该有不少人开发过程中遇到过这样的需求,我们直接看图说话: 如上图所示,其中Entity为实体类,其中包括五个属性,该五个属性的值分别来自于数据库查询结果: 用户通过可视化界面进行 ...
- C#动态表达式计算(续2)
上两篇废话太多,这一次我就不多说了,由于代码比较简单,可以直接从https://github.com/scottshare/DynamicExpress.git地址下载. 以下说明一下使用方法: Dy ...
- 栈应用之 后缀表达式计算 (python 版)
栈应用之 后缀表达式计算 (python 版) 后缀表达式特别适合计算机处理 1. 中缀表达式.前缀表达式.后缀表达式区别 中缀表达式:(3 - 5) * (6 + 17 * 4) / 3 17 ...
- C++实现 逆波兰表达式计算问题
C++实现 逆波兰表达式计算问题 #include <iostream> #include <string> using namespace std; class Stack ...
- 算法笔记_044:表达式计算求值(Java)
目录 1 问题描述 2 解决方案 1 问题描述 问题描述 输入一个只包含加减乖除和括号的合法表达式,求表达式的值.其中除表示整除. 输入格式 输入一行,包含一个表达式. 输出格式 输出这个表达式的 ...
- Fel表达式计算引擎学习
转载原文地址:Fel是轻量级的高效的表达式计算引擎 Fel的问题 Fel的问题 Fel是轻量级的高效的表达式计算引擎 Fel在源自于企业项目,设计目标是为了满足不断变化的功能需求和性能需求. Fel是 ...
- 蓝桥杯 算法训练 ALGO-156 表达式计算
算法训练 表达式计算 时间限制:1.0s 内存限制:256.0MB 问题描述 输入一个只包含加减乖除和括号的合法表达式,求表达式的值.其中除表示整除. 输入格式 输入一行,包含一个表达式. 输 ...
随机推荐
- 2021.08.09 P7238 迷失森林(树的直径)
2021.08.09 P7238 迷失森林(树的直径) P7238 「DCOI」迷失森林 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 重点: 1.树的直径两种求法:两次dfs.树 ...
- 企业级 Web 开发的挑战
本文翻译自土牛Halil ibrahim Kalkan的<Mastering ABP Framework>,是系列翻译的起头,适合ABP开发人员或者想对ABP框架进行深入演进的准架构师. ...
- 1.1 Qt Creater使用Python开发桌面软件的操作流程
Qt Creater及Python的下载与安装过程不再赘述,读者可自行在网上搜索相应的下载与安装方法. 首先我们打开Qt Creater,单击"Create Project"按钮或 ...
- Redis 缓存穿透、缓存击穿、缓存雪崩的解决方案
一.缓存雪崩 缓存雪崩表示:指缓存同一时间大面积失效或缓存重启又或者第一次启用缓存的情况下,导致请求跳过缓存直接请求数据库,造成数据库短时间内承受大量请求而崩掉. 解决方案: 方案一 缓存数据的过期时 ...
- 详解Docker中Image、Container与 Volume 的迁移
开源Linux 长按二维码加关注~ 上一篇:Linux Used内存到底哪里去了? 已经部署的容器化服务,也不是不需要维护的.而且,由于生产环境往往有这样那样的严格要求,往往需要些非常规操作.Imag ...
- linux网络配置,查看IP地址
linux等等学习参考博客:https://www.cnblogs.com/pyyu/p/9276851.html 1.在安装好的linux上面输入cd /etc/sysconfig/network- ...
- selenium模块使用详解、打码平台使用、xpath使用、使用selenium爬取京东商品信息、scrapy框架介绍与安装
今日内容概要 selenium的使用 打码平台使用 xpath使用 爬取京东商品信息 scrapy 介绍和安装 内容详细 1.selenium模块的使用 # 之前咱们学requests,可以发送htt ...
- netty系列之:netty中常用的对象编码解码器
目录 简介 什么是序列化 重构序列化对象 序列化不是加密 使用真正的加密 使用代理 Serializable和Externalizable的区别 netty中对象的传输 ObjectEncoder O ...
- linxu篇-centos搭建ftp服务器
1安装vsftpd 2备份配置文件 3修改配置文件 vi /etc/vsftpd/vsftpd.conf anonymous_enable=NO #允许匿名用户访问为了安全选择关闭 local_ena ...
- 分享一个 SpringCloud Feign 中所埋藏的坑
背景 前段时间同事碰到一个问题,需要在 SpringCloud 的 Feign 调用中使用自定义的 URL:通常情况下是没有这个需求的:毕竟都用了 SpringCloud 的了,那服务之间的调用都是走 ...