【Leetcode】241. 为运算表达式设计优先级

news2024/12/25 0:16:11

241. 为运算表达式设计优先级(中等)

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

解法一:分治法

对于这道题,加括号其实就是决定运算次序,所以我们可以把加括号转化为,「对于每个运算符号,先执行处理两侧的数学表达式,再处理此运算符号」。

对于一个形如 x op y 的算式而言,它的结果组合取决于 x 和 y 的结果组合数 ,而 x 和 y 又可以写成形如 x op y 的算式。

因此,该问题的子问题就是 x op y 中的 x 和 y:以运算符分隔的左右两侧算式解

对于分治算法分成三步走:

  • 分解:按照运算符分隔成左右两部分,分别求解;
  • 解决:通过递归实现,最终我们会得到只包含数字的算式,返回数字作为算式解;
  • 合并:根据运算符合并左右两部分的解,得出最终解。

代码

class Solution {
public:
    vector<int> diffWaysToCompute(string expression) {
        vector<int> ans, left, right;
        int flag = 0; 
        int n = expression.size();
        for(int i=0; i<n; ++i){
            // 如果 expression[i] 是运算符号
            if(!isdigit(expression[i])){
                flag = 1; // flag=1说明string是表达式,flag=0说明string是一个数字
                // string s(str, begin, len):
                // 将字符串str中从下标begin开始、长度为len的部分作为字符串初值
                left = diffWaysToCompute(string(expression, 0, i));
                right = diffWaysToCompute(string(expression, i+1, n-i));
                // 遍历left和right的所有结果数
                for(int l : left){
                    for(int r : right){
                        if(expression[i] == '+') ans.push_back(l + r);
                        if(expression[i] == '-') ans.push_back(l - r);
                        if(expression[i] == '*') ans.push_back(l * r);
                    }
                }
            }
        }
        if(flag == 0){
            // expression 只是一个数字
            // {int} -> vector<int>
            return {stoi(expression)};
        }
        return ans;
    }
};

解法二:分治法+记忆化

解法一中存在重复运算,比如 2*3 - 4*5 ,按照第一个 * 分割,右边部分是 3 - 4*5 ,进而计算 4*5 ,而按照 - 分割,同样会再次计算 4*5

为了减少重复运算,可以使用记忆化搜索,保存字符串区间[l,r]的运算结果,整个思路和解法一类似,不同点在于:设置一个 map 保存结果,递归的时候先在 map 中查找,如果该字符串已经计算过,那么直接返回保存的结果。

代码

class Solution {
public:
    unordered_map<string, vector<int>> mp;
    vector<int> diffWaysToCompute(string expression) {
        if(mp.find(expression) != mp.end()) return mp.find(expression) -> second;
        vector<int> ans, left, right;
        int flag = 0; 
        int n = expression.size();
        for(int i=0; i<n; ++i){
            // 如果 expression[i] 是运算符号
            if(!isdigit(expression[i])){
                flag = 1; // flag=1说明string是表达式,flag=0说明string是一个数字
                // string s(str, begin, len):
                // 将字符串str中从下标begin开始、长度为len的部分作为字符串初值
                left = diffWaysToCompute(string(expression, 0, i));
                right = diffWaysToCompute(string(expression, i+1, n-i));
                // 遍历left和right的所有结果数
                for(int l : left){
                    for(int r : right){
                        if(expression[i] == '+') ans.push_back(l + r);
                        if(expression[i] == '-') ans.push_back(l - r);
                        if(expression[i] == '*') ans.push_back(l * r);
                    }
                }
            }
        }
        if(flag == 0){
            // expression 只是一个数字
            // {int} -> vector<int>
            return {stoi(expression)};
        }
        mp[expression] = ans;
        return ans;
    }
};

解法三:动态规划

对于表达式 expression 需要做预处理:「把每个数字转为 int 存起来,同时运算符也存起来」。

这样子将得到两个数组,以 2 * 3 - 4 * 5 为例,存起来的数字是 numList = [2 3 4 5],存起来的运算符是 opList = [*, -, *]

状态定义

dp[i][j] 表示从第 i 个数字到第 j 个数字(从 0 开始计数)范围内表达式的所有解。

dp[1][3]表示:第 1 个数字 (3) 到 第 3 个数字(5)范围内表达式 3 - 4 * 5 的所有解。

状态转移方程

有了一个数字的所有解(初始化),就可以求出两个数字的所有解。

有了两个数字的所有解,三个数字的所有解就和解法一求法一样。

把三个数字分成两部分,将两部分的解两两组合起来即可。

对于两部分之间的运算符,因为表达式是一个数字一个运算符,所以运算符的下标就是左部分最后一个数字的下标

对于 2 * 3 - 4 * 5 ,存起来的数字是 numList = [2 3 4 5], 存起来的运算符是 opList = [*, -, *]

假设要求 dp[1][3],也就是计算 3 - 4 * 5 的解:

  • 分成 3 和 4 * 5 两部分,3 对应的下标是 1 ,对应的运算符就是 opList[1] = '-' ,也就是计算 3 - 20 = -17

  • 分成 3 - 4 和 5 两部分,4 的下标是 2 ,对应的运算符就是 opList[2] = '*'
    也就是计算 -1 * 5 = -5

  • 所以 dp[1][3] = [-17 -5]

四个、五个… 都可以分成两部分,然后通过之前的解求出来。

直到包含了所有数字的解求出来。

初始化

对范围内只有一个数字的情况进行初始化:

dp[0][0] = 2, dp[1][1] = 3, dp[2][2] = 4, dp[3][3] = 5;

返回的最终结果

最终返回第 0 个数字到最后一个数字范围内表达式的所有解,即 dp[0][n-1]

代码

class Solution {
public:
    vector<int> diffWaysToCompute(string expression) {
        vector<int>  numList;
        vector<char> opList;
        
        // 对 expression 预处理
        int num = 0;
        for(char ch : expression){
            if(!isdigit(ch)){
                numList.push_back(num);
                num = 0;
                opList.push_back(ch);
            }
            else{
                num = num * 10 + ch - '0';
            }
        }
        numList.push_back(num);
        int n = numList.size();
        vector<vector<vector<int>>> dp(n, vector<vector<int>>(n));

        // dp数组初始化
        for(int i=0; i<n; ++i){
            dp[i][i].push_back(numList[i]);
        }
        // 从2个数字遍历到n个数字
        for(int k=2; k<=n; ++k){
            // 开始遍历的下标
            for(int i=0; i<n; ++i){
                // 结束遍历的下标
                int j = i + k - 1;
                if(j >= n){
                    // 越界
                    break;
                }

                vector<int> ans;
                // 以s作为分隔点分成左右两部分遍历
                for(int s=i; s<j; ++s){
                    vector<int> left = dp[i][s];
                    vector<int> right = dp[s+1][j];
                    // 遍历左边部分的所有结果值
                    for(auto x : left){
                        // 遍历右边部分的所有结果值
                        for(auto y : right){
                            // 根据操作符对所有结果进行组合
                            // 操作符下标就是左边部分最后一个数字的下标
                            char op = opList[s];
                            if(op == '+') ans.push_back(x + y);
                            else if(op == '-') ans.push_back(x - y);
                            else if(op == '*') ans.push_back(x * y);
                        }
                    }
                }
                dp[i][j] = ans;
            }
        }
        return dp[0][n-1];
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/507243.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

提高APP安全性的必备加固手段——深度解析代码混淆技术

APP 加固方式 Android APP 加固是优化 APK 安全性的一种方法&#xff0c;常见的加固方式有混淆代码、加壳、数据加密、动态加载等。下面介绍一下 Android APP 加固的具体实现方式。 混淆代码&#xff1a; 使用 ProGuard 工具可以对代码进行混淆&#xff0c;使得反编译出来的代…

makefile语法解析

gcc语法简介 gcc (-c) hello.c world.c -o main-c表示只编译不链接 -o目标文件 此外&#xff0c;还有-I&#xff08;大写的I&#xff09;表示给gcc添加自定义的头文件的路径 -L表示给gcc添加额外的搜索库的路径 -g选项的意义是“生成调试信息&#xff0c;该程序可以被调试器调…

【Java入门合集】第五章抽象类和接口(二)

博主&#xff1a;命运之光 专栏&#xff1a;JAVA入门 学习目标 1.了解什么是抽象类&#xff0c;什么是接口&#xff1b; 2.掌握抽象类和接口的定义方法&#xff1b; 3.理解接口和抽象类的使用场景&#xff1b; 4.掌握多态的含义和用法&#xff1b; 5.掌握内部类的定义方法和使用…

【Hive实战】Hive元数据存储库数据增多的分析

Hive元数据存储库数据增多的分析 2023年5月8日 文章目录 Hive元数据存储库数据增多的分析问题新增Hive相关的DDL操作创建Hive库库授权到用户 创建Hive表 内部表非分区表表授权到用户一级分区表二级分区表分桶表分桶排序表 查询指令核心表分析表关系图表以库表为主以hive表为主以…

人人都可用的ChatGPT,Edge浏览器-免费ChatGPT保姆级教程!非常详细!

人工智能大浪潮已经来临&#xff0c;对于ChatGPT&#xff0c;我觉得任何一个玩互联网的人&#xff0c;都应该重视起来&#xff0c;用起来。但是国内使用需要解决科学上网、注册、收费等繁琐问题。 所以&#xff0c;今天这篇文章就来推荐一个插件&#xff0c;无需任何繁琐操作&…

第一次省赛团队训练 - BAPC 2022 Pre(DAPC 2022)

B (2). Bubble-bubble Sort [ 提交记录 ] [ 问题 3462 ] 时间限制: 2000MS 空间限制: 256MB 结果评判: 文本对比 正确/提交: 7 (5) / 15 官方标签: 用户标记: 题目描述 Bubbles! As a fanatical supporter of the Bubbles Are Perfect Creatures movement, you have ac…

第8章 未执行缓存的强制清理操作导致显示异常解决方案

1 异常产生原因&#xff1a; 由于未为Role实体定义相就的缓存强制销毁器类&#xff1a;Services.Customers.Caching.RoleCacheEventConsumer,从而导致Services.Events.EventPublisher.PublishAsync<TEvent>(TEvent event)中的 consumers实例为0,如下图所示&#xff1a; 2…

深入理解移动端布局:Viewport与设备像素比

在移动端开发中&#xff0c;了解和掌握不同设备的布局特点是非常重要的。本文将介绍两个关键概念&#xff1a;Viewport 和设备像素比&#xff08;DPR&#xff09;&#xff0c;帮助你更好地理解移动端布局。 一、什么是 Viewport&#xff1f; Viewport 是用户在浏览器中可见的网…

JS知识点(包括原型,原型对象,数据类型,数据类型的检测)

目录 1、JavaScript有哪些数据类型&#xff0c;它们的区别&#xff1f; 2、基本数据类型和引用数据类型地区别&#xff1a; 3、数据类型检测的方式有哪些: 4、判断数组的方式有那些&#xff1f; 5、null和undefined区别&#xff1a; 6、为什么typeOf null得到object而不是n…

22个提升生产力的工具推荐,稳了

子曰&#xff1a;工欲善其事&#xff0c;必先利其器。 本文给大家推荐22个提高生产力的工具&#xff0c;总有一款符合你的需求。&#x1f604;&#x1f604;&#x1f604; 提高生产效率工具推荐 滴答清单/Todoist文件检索利器&#xff1a;Everything文件管理软件-Allen Explor…

基于SpringBoot的大学生租房系统

背景 大学生租房系统设计的目的是建立一个高效的平台&#xff0c;采用简洁高效的Java语言与Mysql数据库等技术&#xff0c;设计和开发了本大学生租房系统设计。该系统主要实现了用户和房主通过系统注册用户&#xff0c;登录系统后能够编辑自己的个人信息、查看首页&#xff0c…

【电子学会】2023年03月图形化三级 -- 猫猫的儿童节

猫猫的儿童节 儿童节到了&#xff0c;给小猫绘制一个七彩的气球。 1. 准备工作 &#xff08;1&#xff09;保留小猫角色&#xff1b; &#xff08;2&#xff09;选择“Button2”角色&#xff0c;添加文字“开始”&#xff1b; &#xff08;3&#xff09;默认白色背景。 2…

有人抱怨Android找不到工作,有人却收到了好几个Offer~

不知不觉&#xff0c;往年常说的面试黄金季就这样过去了&#xff0c;相信现在很多人都会抱怨说&#xff0c;现在是市场岗位缩水裁员季。有人抱怨&#xff0c;自然也有人喜悦&#xff0c;有失业人群在&#xff0c;自然就业人群也有&#xff0c;有人想找一份合理工作很难&#xf…

C高级(day1)

作业: 初始工作路径不在家目录下&#xff0c;在不切换路径的情况下&#xff0c;在家目录下创建一个subdir目录&#xff0c;在subdir这个目录下&#xff0c;创建subdir1和subdir2&#xff0c;并且把/etc/passwd拷贝到subdir1中&#xff0c;把/etc/group文件拷贝到subdir2中&…

David Silver Lecture 5: Model-Free Control

1 Introduction 1.1 内容 上一章是对一个unknown MDP进行value function的预测&#xff0c;相当于policy evaluation。这一章是对unknown MDP找到一个最优的policy&#xff0c; optimise value function. 1.2 On and Off-Policy Learning On-policy learning learn on the…

[oeasy]python0050_动态类型_静态类型_编译_运行

动态类型_静态类型 回忆上次内容 上次了解了 帮助文档的 生成 开头的三引号注释 可以生成 帮助文档文档 可以写成网页 python3 本身 也有 在线的帮助手册 目前的程序 提高了 可读性 有什么方法 可以让程序 更可读么&#xff1f;&#x1f914; 变量名 首先 在变量名上想办…

opencv_c++学习(六)

一、视频加载与摄像头调用 视频、摄像头加载 VideoCapture(filename, CAP_ANY)对以上实例解释如下&#xff1a; 若读取的为本地视频&#xff0c;则filename为视频名称&#xff0c;若读取的是摄像头数据&#xff0c;则为int类型的摄像头id。 视频属性的获取 视频属性可以通过…

手握美团offer,结果背调红灯,哭了....

相信很多人都会包装简历&#xff0c;尤其是工作经历&#xff0c;不过也有人会填一下虚假的背景信息&#xff0c;比如公司leader或HR&#xff0c;小公司没有实力过多进行背调&#xff0c;但是大企业就不同了&#xff0c;他们有方法了解到实际的情况。 背调包括候选人以往的经历…

RHCSA之Linux的安装步骤

目录 RHCSA之环境配置 需要的软件 VMwareWorkstation安装 1.打开VMwareWorkstation安装包 2.进入安装界面点击下一步 3. 在我接受许可协议打 √ 后&#xff0c;点击下一步 4.在安装位置选择更改 5. 更改目标安装位置&#xff0c;点击确定 6.疯狂点击下一步 8.点击安装 9.…

DDIM模型代码实现

背景 前面已经出了一系列的文章来介绍大模型、多模态、生成模型。这篇文章会从更微观和更贴近实际工作的角度下手。会给大家介绍下前面讲到的diffuiosn model具体怎么来实现。文章结构如下&#xff1a; 1.介绍Diffusion Model包括哪些零部件&#xff0c;这些零部件衔接关系 …