Go语言学习编程实践:实现简易计算器(包含词法器、语法树构建)

简介: Go语言学习编程实践:实现简易计算器(包含词法器、语法树构建)

1.png

实现流程:

①把input输入的字符串进行拆分,变成一个一个单词,要设计词法分析器,大部分都设计相关的词法分析器。parser。它会不断读取每一个字符,然后生成对应的一个个词元。

我们将词元和词法分析器,分别用两个结构体表示出来。

type Token struct {
  Type string  //对应我们上面的词元类型
  Literal string // 实际的string字符
}
type Lexer struct {
  input string // 输入
  position int   // 当前位置                                                                                                                                                                                                                                                                                                                                                                                               
  readPosition int  // 将要读取的位置
  ch byte //当前字符
}

词法分析器操作:

  • 不断读取字符串的每一个字符,更新结构体的值。
  • 要注意考虑跳过空白字符/制表符。
  • 对于数字要单独考虑,因为 ’567‘ 这是一个词元。

②构建语法树,要让计算机按照你想要的方式进行计算。遵循数学里面的计算法则。之前数据结构中学习二叉树时,都知道中缀表达式,对应了我们的计算规则。当然前缀/后缀也可以计算出正确的值,比如逆波兰式求职,这里不在赘述。可以看另一篇博客有通过后缀表达式计算值,安置传送门。

③考虑两种计算情况。

情况一:++i;某个操作符作为前缀与后面数字发生反应。同样还包括我们的-1。

情况二:1 + 2。操作符在中间。

// package calc
package main
import (
  "bufio"
  "bytes"
  "fmt"
  "os"
  "strconv"
)
const (
  ILLEGAL = "ILLEGAL"
  EOF     = "EOF"
  INT     = "INT"
  PLUS     = "+"
  MINUS    = "-"
  ASTERISK = "*"
  SLASH    = "/"
  LPAREN = "("
  RPAREN = ")"
)
const (
  _ int = iota
  LOWEST
  SUM     // +, -
  PRODUCT // *, /
  PREFIX  // -X
  CALL    // (X)
)
var precedences = map[string]int{
  PLUS:     SUM,
  MINUS:    SUM,
  SLASH:    PRODUCT,
  ASTERISK: PRODUCT,
  LPAREN:   CALL,
}
func Calc(input string) int64 {
  lexer := NewLex(input)
  parser := NewParser(lexer)
  exp := parser.ParseExpression(LOWEST)
  return Eval(exp)
}
func main() {
  reader := bufio.NewReader(os.Stdin)
  n, err := reader.ReadString('\n')
  if err != nil {
    fmt.Println("error")
  }
  //fmt.Println(n)
  fmt.Println(Calc(n))
}
func Eval(exp Expression) int64 {
  switch node := exp.(type) {
  case *IntegerLiteralExpression:
    return node.Value
  case *PrefixExpression:
    rightV := Eval(node.Right)
    return evalPrefixExpression(node.Operator, rightV)
  case *InfixExpression:
    leftV := Eval(node.Left)
    rightV := Eval(node.Right)
    return evalInfixExpression(leftV, node.Operator, rightV)
  }
  return 0
}
func evalPrefixExpression(operator string, right int64) int64 {
  if operator != "-" {
    return 0
  }
  return -right
}
func evalInfixExpression(left int64, operator string, right int64) int64 {
  switch operator {
  case "+":
    return left + right
  case "-":
    return left - right
  case "*":
    return left * right
  case "/":
    if right != 0 {
      return left / right
    } else {
      return 0
    }
  default:
    return 0
  }
}
type Token struct {
  Type    string
  Literal string
}
func newToken(tokenType string, c byte) Token {
  return Token{
    Type:    tokenType,
    Literal: string(c),
  }
}
type Lexer struct {
  input        string
  position     int
  readPosition int
  ch           byte
}
func NewLex(input string) *Lexer {
  l := &Lexer{input: input}
  l.readChar()
  return l
}
func (l *Lexer) NextToken() Token {
  var tok Token
  l.skipWhitespace() //跳过全部空白字符或制表符
  switch l.ch {
  case '(':
    tok = newToken(LPAREN, l.ch)
  case ')':
    tok = newToken(RPAREN, l.ch)
  case '+':
    tok = newToken(PLUS, l.ch)
  case '-':
    tok = newToken(MINUS, l.ch)
  case '/':
    tok = newToken(SLASH, l.ch)
  case '*':
    tok = newToken(ASTERISK, l.ch)
  case 0: //对于超出字符串范围的进行结束 EOF
    tok.Literal = ""
    tok.Type = EOF
  default:
    if isDigit(l.ch) {
      tok.Type = INT
      tok.Literal = l.readNumber()
      return tok
    } else {
      tok = newToken(ILLEGAL, l.ch) //非法字符处理
    }
  }
  l.readChar()
  return tok
}
func isDigit(ch byte) bool {
  return '0' <= ch && ch <= '9'
}
// 读一个字符,超出输入的字符串范围,就将当前字符置为0,结束读取
func (l *Lexer) readChar() {
  if l.readPosition >= len(l.input) {
    l.ch = 0
  } else {
    l.ch = l.input[l.readPosition]
  }
  l.position = l.readPosition
  l.readPosition += 1
}
// 读取数字的单独函数 因为345这是一个数
func (l *Lexer) readNumber() string {
  position := l.position
  for isDigit(l.ch) {
    l.readChar()
  }
  return l.input[position:l.position] //返回一个数据
}
// 知道是这些制表符就再往下读一个
func (l *Lexer) skipWhitespace() {
  for l.ch == ' ' || l.ch == '\t' || l.ch == '\n' || l.ch == '\r' {
    l.readChar()
  }
}
// ast
type Expression interface {
  String() string
}
type IntegerLiteralExpression struct {
  Token Token
  Value int64
}
func (il *IntegerLiteralExpression) String() string { return il.Token.Literal }
type PrefixExpression struct {
  Token    Token
  Operator string
  Right    Expression
}
func (pe *PrefixExpression) String() string {
  var out bytes.Buffer
  out.WriteString("(")
  out.WriteString(pe.Operator)
  out.WriteString(pe.Right.String())
  out.WriteString(")")
  return out.String()
}
type InfixExpression struct {
  Token    Token
  Left     Expression
  Operator string
  Right    Expression
}
func (ie *InfixExpression) String() string {
  var out bytes.Buffer
  out.WriteString("(")
  out.WriteString(ie.Left.String())
  out.WriteString(" ")
  out.WriteString(ie.Operator)
  out.WriteString(" ")
  out.WriteString(ie.Right.String())
  out.WriteString(")")
  return out.String()
}
// parser
type (
  prefixParseFn func() Expression //函数 返回参数是Expression接口
  infixParseFn  func(Expression) Expression
)
type Parser struct {
  l *Lexer
  curToken  Token
  peekToken Token
  prefixParseFns map[string]prefixParseFn //map的值是函数
  infixParseFns  map[string]infixParseFn
  errors []string
}
// 往结构体里面筛方法
func (p *Parser) registerPrefix(tokenType string, fn prefixParseFn) {
  p.prefixParseFns[tokenType] = fn
}
func (p *Parser) registerInfix(tokenType string, fn infixParseFn) {
  p.infixParseFns[tokenType] = fn
}
func NewParser(l *Lexer) *Parser {
  p := &Parser{
    l:      l,
    errors: []string{},
  }
  //这个结构体加入三种方法
  p.prefixParseFns = make(map[string]prefixParseFn)
  p.registerPrefix(INT, p.parseIntegerLiteral)
  p.registerPrefix(MINUS, p.parsePrefixExpression)
  p.registerPrefix(LPAREN, p.parseGroupedExpression)
  //这个加入四种方法 + - * /
  p.infixParseFns = make(map[string]infixParseFn)
  p.registerInfix(PLUS, p.parseInfixExpression)
  p.registerInfix(MINUS, p.parseInfixExpression)
  p.registerInfix(SLASH, p.parseInfixExpression)
  p.registerInfix(ASTERISK, p.parseInfixExpression)
  p.nextToken()
  p.nextToken()
  return p
}
func (p *Parser) ParseExpression(precedence int) Expression {
  prefix := p.prefixParseFns[p.curToken.Type]
  returnExp := prefix()
  for precedence < p.peekPrecedence() {
    infix := p.infixParseFns[p.peekToken.Type]
    if infix == nil {
      return returnExp
    }
    p.nextToken()
    returnExp = infix(returnExp)
  }
  return returnExp
}
func (p *Parser) peekPrecedence() int {
  if p, ok := precedences[p.peekToken.Type]; ok {
    return p
  }
  return LOWEST
}
func (p *Parser) curPrecedence() int {
  if p, ok := precedences[p.curToken.Type]; ok {
    return p
  }
  return LOWEST
}
func (p *Parser) peekError(t string) {
  msg := fmt.Sprintf("expected next token to be %s, got %s instend",
    t, p.peekToken.Type)
  p.errors = append(p.errors, msg)
}
func (p *Parser) expectPeek(t string) bool {
  if p.peekTokenIs(t) {
    p.nextToken()
    return true
  } else {
    p.peekError(t)
    return false
  }
}
func (p *Parser) peekTokenIs(t string) bool {
  return p.peekToken.Type == t
}
func (p *Parser) nextToken() {
  p.curToken = p.peekToken
  p.peekToken = p.l.NextToken()
}
func (p *Parser) parseIntegerLiteral() Expression {
  lit := &IntegerLiteralExpression{Token: p.curToken}
  value, err := strconv.ParseInt(p.curToken.Literal, 0, 64)
  if err != nil {
    msg := fmt.Sprintf("could not parse %q as integer", p.curToken.Literal)
    p.errors = append(p.errors, msg)
    return nil
  }
  lit.Value = value
  return lit
}
func (p *Parser) parsePrefixExpression() Expression {
  expression := &PrefixExpression{
    Token:    p.curToken,
    Operator: p.curToken.Literal,
  }
  p.nextToken()
  expression.Right = p.ParseExpression(PREFIX)
  return expression
}
func (p *Parser) parseGroupedExpression() Expression {
  p.nextToken()
  exp := p.ParseExpression(LOWEST)
  if !p.expectPeek(RPAREN) {
    return nil
  }
  return exp
}
func (p *Parser) parseInfixExpression(left Expression) Expression {
  expression := &InfixExpression{
    Token:    p.curToken,
    Operator: p.curToken.Literal,
    Left:     left,
  }
  precedence := p.curPrecedence()
  p.nextToken()
  // // 通过降低优先级,来达到右结合
  //if expression.Operator == "+" {
  //  expression.Right = p.parseExpression(precedence - 1)
  //} else {
  //  expression.Right = p.parseExpression(precedence)
  //}
  expression.Right = p.ParseExpression(precedence)
  return expression
}
func (p *Parser) Errors() []string {
  return p.errors
}
相关文章
|
11天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
52 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
30天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
106 67
|
6天前
|
算法 安全 Go
Go 语言中实现 RSA 加解密、签名验证算法
随着互联网的发展,安全需求日益增长。非对称加密算法RSA成为密码学中的重要代表。本文介绍如何使用Go语言和[forgoer/openssl](https://github.com/forgoer/openssl)库简化RSA加解密操作,包括秘钥生成、加解密及签名验证。该库还支持AES、DES等常用算法,安装简便,代码示例清晰易懂。
34 12
|
9天前
|
监控 算法 安全
解锁企业计算机监控的关键:基于 Go 语言的精准洞察算法
企业计算机监控在数字化浪潮下至关重要,旨在保障信息资产安全与高效运营。利用Go语言的并发编程和系统交互能力,通过进程监控、网络行为分析及应用程序使用记录等手段,实时掌握计算机运行状态。具体实现包括获取进程信息、解析网络数据包、记录应用使用时长等,确保企业信息安全合规,提升工作效率。本文转载自:[VIPShare](https://www.vipshare.com)。
19 0
|
23天前
|
Go 数据安全/隐私保护 UED
优化Go语言中的网络连接:设置代理超时参数
优化Go语言中的网络连接:设置代理超时参数
|
1月前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
41 7
|
1月前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
1月前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
102 71
|
1月前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
102 62
|
1月前
|
存储 Go
go语言中映射
go语言中映射
38 11