736. Lisp语法解析

题目描述:

给你一个类似 Lisp 语句的字符串表达式 expression,求出其计算结果。

表达式语法如下所示:

  • 表达式可以为整数,let 表达式,add 表达式,mult 表达式,或赋值的变量。表达式的结果总是一个整数。
  • (整数可以是正整数、负整数、0)
  • let 表达式采用 “(let v1 e1 v2 e2 … vn en expr)” 的形式,其中 let 总是以字符串 “let”来表示,接下来会跟随一对或多对交替的变量和表达式,也就是说,第一个变量 v1被分配为表达式 e1 的值,第二个变量 v2 被分配为表达式 e2 的值,依次类推;最终 let 表达式的值为 expr表达式的值。
  • add 表达式表示为 “(add e1 e2)” ,其中 add 总是以字符串 “add” 来表示,该表达式总是包含两个表达式 e1、e2 ,最终结果是 e1 表达式的值与 e2 表达式的值之 和 。
  • mult 表达式表示为 “(mult e1 e2)” ,其中 mult 总是以字符串 “mult” 表示,该表达式总是包含两个表达式 e1、e2,最终结果是 e1 表达式的值与 e2 表达式的值之 积 。
  • 在该题目中,变量名以小写字符开始,之后跟随 0 个或多个小写字符或数字。为了方便,“add” ,“let” ,“mult” 会被定义为 “关键字” ,不会用作变量名。
  • 最后,要说一下作用域的概念。计算变量名所对应的表达式时,在计算上下文中,首先检查最内层作用域(按括号计),然后按顺序依次检查外部作用域。测试用例中每一个表达式都是合法的。有关作用域的更多详细信息,请参阅示例。

测试用例:

示例 1:

输入:expression = "(let x 2 (mult x (let x 3 y 4 (add x y))))"
输出:14
解释:
计算表达式 (add x y), 在检查变量 x 值时,
在变量的上下文中由最内层作用域依次向外检查。
首先找到 x = 3, 所以此处的 x 值是 3 。

示例 2:

输入:expression = "(let x 3 x 2 x)"
输出:2
解释:let 语句中的赋值运算按顺序处理即可。

示例 3:

输入:expression = "(let x 1 y 2 x (add x y) (add x y))"
输出:5
解释:
第一个 (add x y) 计算结果是 3,并且将此值赋给了 x 。
第二个 (add x y) 计算结果是 3 + 2 = 5 。

限制及提示:

  • 1 <= expression.length <= 2000
  • exprssion 中不含前导和尾随空格
  • expressoin 中的不同部分(token)之间用单个空格进行分隔
  • 答案和所有中间计算结果都符合 32-bit 整数范围
  • 测试用例中的表达式均为合法的且最终结果为整数

解题分析及思路:

本题的难点在于思路以及一些细节的处理。

首先,对于一个表达式他一定是( + expression + )的形式,而潜逃的表达式也同样是多个此类结构的组合,那么可以将括号单独判断,直接忽略括号,重点处理里面的expression即可。

对于一个纯粹的expression,即不包括嵌套类型的表达式,他一定由变量、整数、let、add、mult组成,而let、add、mult一定是紧跟(,所以只要判断是(之后,如果立即以l、a、m开头,即可进行let、add、mult的运算。

对于let、add、mult三种运算:

  1. let:可能会有以下情况:

    • 如果是变量与表达式的交替,我们需要将其变量的值以栈的形式保存起来。
    • 如果下一个字符不是小写字母,或者解析变量后下一个字符是右括号 ),说明是最后一个表达式,计算它的值并返回结果。并且我们需要在 scope 中清除 let 表达式对应的作用域变量。
  2. add:后面会紧跟两个表达式,计算两个表达式的和即可

  3. mult:后面紧跟两个表达式,计算两个表达式的乘积即可

对于变量和整数的解析:

  • 变量:从其实位置遇到空格或者)即为变量的全部。
  • 整数:如果为数字,临时值*10 + 该位置数字在存入临时值,遇到非数字停止解析,可能为负数,在解析钱额外判断是否以-开头即可。

对于括号内可能仍然以多个括号的形式,不正是和递归一致吗,所以遇到(进行递归即可。

代码分析:

解析变量:

func() string {
    i0 := i
    for i < n && expression[i] != ' ' && expression[i] != ')' {
        i++
    }
    return expression[i0:i]
}

解析整数:

func() int {
    sign, x := 1, 0
    // 判断正负
    if expression[i] == '-' {
        sign = -1
        i++
    }
    // 计算数值
    for i < n && unicode.IsDigit(rune(expression[i])) {
        x = x*10 + int(expression[i]-'0')
        i++
    }
    return sign * x
}

计算let

// 移除 "let "
i += 4
var vars []string
for {
    if !unicode.IsLower(rune(expression[i])) {
        ret = innerEvaluate() // let 表达式的最后一个 expr 表达式的值
        break
    }
    vr := parseVar()
    if expression[i] == ')' {
        vals := scope[vr]
        ret = vals[len(vals)-1] // let 表达式的最后一个 expr 表达式的值
        break
    }
    vars = append(vars, vr)
    i++ // 移除空格
    scope[vr] = append(scope[vr], innerEvaluate())
    i++ // 移除空格
}
for _, vr := range vars {
    scope[vr] = scope[vr][:len(scope[vr])-1] // 清除当前作用域的变量
}

计算add:

// 移除 "add "
i += 4
e1 := innerEvaluate()
// 移除空格
i++
e2 := innerEvaluate()
ret = e1 + e2

计算mult:

// 移除 "mult "
i += 5
e1 := innerEvaluate()
// 移除空格
i++
e2 := innerEvaluate()
ret = e1 * e2

复杂度:

  • 时间复杂度:O(n)
  • 空间复杂度:O(n)

执行结果:

  • 执行用时: 0 ms , 在所有 Go 提交中击败了 100.00% 的用户
  • 内存消耗: 2.5 MB , 在所有 Go 提交中击败了 76.19% 的用户
Tags :

通过次数 14.2K 提交次数 21.6K 通过率 65.9%

Related Posts

1206. 设计跳表

## 题目描述:不使用任何库函数,设计一个 跳表 。跳表 是在 O(log(n)) 时间内完成增加、删除、搜索操作的数据结构。跳表相比于树堆与红黑树,其功能与性能相当,并且跳表的代码长度相较下更短,其设计思想与链表相似。例如,一个跳表包含 [30, 40, 50, 60, 70, 90] ,然后增加 80、45 到跳表中,以下图的方式操作。![](/img/leetcode/1

read more

41. 缺失的第一个正数

## 题目描述:给你一个未排序的整数数组 nums ,请你找出其中没有出现的最小的正整数。请你实现时间复杂度为 O(n) 并且只使用常数级别额外空间的解决方案。示例 1:输入:nums = [1,2,0] 输出:3 解释:范围 [1,2] 中的数字都在数组中。示例 2:``` 输入:nums = [3,4,-1,1] 输出:2 解释

read more

42. 接雨水

## 题目描述:给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。示例 1: 输入:height = [0,1,0,2,1,0,1,3,2,1,2,1] 输出:6 解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。 ![

read more

899. 有序队列

## 题目描述:给定一个字符串 s 和一个整数 k 。你可以从 s 的前 k 个字母中选择一个,并把它加到字符串的末尾。返回 在应用上述步骤的任意数量的移动后,字典上最小的字符串 。**测试用例:**示例 1: ``` 输入:s = "cba", k = 1 输出:"acb" 解释: 在第一步中,我们将第一个字符(“c”)移动到最后,获得字符串 “bac”。 在第二步中,我们将第

read more