[Leetcode] 770. Basic Calculator IV 解题报告

本文介绍LeetCode第770题——基本计算器IV的解题思路和解决方案。该题目要求根据输入表达式和评估映射返回简化后的表达式列表,遵循特定的格式和运算顺序。文章详细解释了解题过程,并提供了代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目

Given an expression such as expression = "e + 8 - a + 5" and an evaluation map such as {"e": 1} (given in terms of evalvars = ["e"] and evalints = [1]), return a list of tokens representing the simplified expression, such as ["-1*a","14"]

  • An expression alternates chunks and symbols, with a space separating each chunk and symbol.
  • A chunk is either an expression in parentheses, a variable, or a non-negative integer.
  • A variable is a string of lowercase letters (not including digits.) Note that variables can be multiple letters, and note that variables never have a leading coefficient or unary operator like "2x" or "-x".

Expressions are evaluated in the usual order: brackets first, then multiplication, then addition and subtraction. For example, expression = "1 + 2 * 3" has an answer of ["7"].

The format of the output is as follows:

  • For each term of free variables with non-zero coefficient, we write the free variables within a term in sorted order lexicographically. For example, we would never write a term like "b*a*c", only "a*b*c".
  • Terms have degree equal to the number of free variables being multiplied, counting multiplicity. (For example, "a*a*b*c" has degree 4.) We write the largest degree terms of our answer first, breaking ties by lexicographic order ignoring the leading coefficient of the term.
  • The leading coefficient of the term is placed directly to the left with an asterisk separating it from the variables (if they exist.)  A leading coefficient of 1 is still printed.
  • An example of a well formatted answer is ["-2*a*a*a", "3*a*a*b", "3*b*b", "4*a", "5*c", "-6"] 
  • Terms (including constant terms) with coefficient 0 are not included.  For example, an expression of "0" has an output of [].

Examples:

Input: expression = "e + 8 - a + 5", evalvars = ["e"], evalints = [1]
Output: ["-1*a","14"]

Input: expression = "e - 8 + temperature - pressure",
evalvars = ["e", "temperature"], evalints = [1, 12]
Output: ["-1*pressure","5"]

Input: expression = "(e + 8) * (e - 8)", evalvars = [], evalints = []
Output: ["1*e*e","-64"]

Input: expression = "7 - 7", evalvars = [], evalints = []
Output: []

Input: expression = "a * b * c + b * a * c * 4", evalvars = [], evalints = []
Output: ["5*a*b*c"]

Input: expression = "((a - b) * (b - c) + (c - a)) * ((a - b) + (b - c) * (c - a))",
evalvars = [], evalints = []
Output: ["-1*a*a*b*b","2*a*a*b*c","-1*a*a*c*c","1*a*b*b*b","-1*a*b*b*c","-1*a*b*c*c","1*a*c*c*c","-1*b*b*b*c","2*b*b*c*c","-1*b*c*c*c","2*a*a*b","-2*a*a*c","-2*a*b*b","2*a*c*c","1*b*b*b","-1*b*b*c","1*b*c*c","-1*c*c*c","-1*a*a","1*a*b","1*a*c","-1*b*c"]

Note:

  1. expression will have length in range [1, 250].
  2. evalvars, evalints will have equal lengths in range [0, 100].

思路

很烦这种题目。。。

代码

class Solution {
public:
    vector<string> basicCalculatorIV(string expression, vector<string>& evalvars, vector<int>& evalints) {
        unordered_map<string, int> eval;                // construct the map from evalvars to evalints
        for (int i = 0; i < evalvars.size(); ++i) {
            eval[evalvars[i]] = evalints[i];
        }
        auto resmp = helper(expression, eval);          // solve the problem
        vector<string> res;                             // construct the results
        for (auto param : resmp) {
            if (param.second == 0) {                    // skip zero coefficient
                continue;
            }
            res.push_back(to_string(param.second));
            for (auto p : param.first) {
                res.back() += "*" + p;
            }
        }
        return res;
    }
private:
    struct cmp {
        bool operator()(vector<string> x, vector<string> y) {   // sort terms by degree
            if (x.size() == y.size()) {
                return x < y;
            }
            else {
                return x.size() > y.size();
            }
        }
    };
    map<vector<string>, int, cmp> helper(string &expr, unordered_map<string, int> &eval) {
        map<vector<string>, int, cmp> local = {{{}, 1}}, global;
        string sub = "";
        int sign = 1, n = expr.size();
        for (int i = 0; i <= n; ++i) {
            if (i < n && expr[i] == ' ') {          // skip space
                continue;  
            }
            if (i < n && isalnum(expr[i])) {        // number and alphabet, add to sub expression
                sub += expr[i];
            }
            else if (i < n && expr[i] == '(') {     // parenthesis
                ++i;
                for (int p = 1; ; ++i) {            // get the sub expression in parenthesis
                    if (expr[i] == '(') {
                        ++p;
                    }
                    else if (expr[i] == ')') {
                        --p;
                    }
                    if (p == 0) {
                        break;
                    }
                    sub += expr[i];
                }
            }
            else {                                  // '+', '-', '*' or i == n
                if (sub.size() == n) {              // the whole expression is a number or variable
                    if (eval.count(sub)) {
                        return {{{}, eval[sub]}};   // variable is found in eval list 
                    }
                    if (isdigit(sub[0])) {
                        return {{{}, stoi(sub)}};   // number
                    }
                    return {{{sub}, 1}};            // variable
                }
                map<vector<string>, int, cmp> mult, submp = helper(sub, eval);
                for (auto l : local) {              // multiply local with sub
                    for (auto r : submp) {
                        auto k = l.first;
                        k.insert(k.end(), r.first.begin(), r.first.end());
                        sort(k.begin(), k.end());
                        mult[k] += l.second * r.second;
                    }
                }
                local = move(mult);
                if (i == n || expr[i] != '*') {     // '+' or '-'
                    for (auto& t : local) {         // add local to global
                        global[t.first] += sign * t.second;
                    }
                    sign = i < n && expr[i] == '-' ? -1 : 1;
                    local = {{{}, 1}};              // reset local
                }
                sub = "";                           // reset sub
            }
        }
        return global;
    }
};


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值