AST 语法树说明

在Go语言中,AST(Abstract Syntax Tree)即抽象语法树。它是一种用于表示源代码结构的数据结构,通过对源代码的语法分析,可以生成一棵AST,用于表示源代码的语法结构。

go/ast

例如,下面是一段Go语言源代码:

package main

import "fmt"

func main() {
    fmt.Println("Hello, world!")
}

go/astparser.ParseFile
package main

import (
    "fmt"
    "go/ast"
    "go/parser"
    "go/token"
)

func main() {
    // 解析源文件
    fset := token.NewFileSet()
    f, err := parser.ParseFile(fset, "example.go", nil, 0)
    if err != nil {
        panic(err)
    }

    // 遍历AST
    ast.Inspect(f, func(n ast.Node) bool {
        // 处理节点
        fmt.Println(n)
        return true
    })
}

局部输出

&{<nil> 1 main [0xc0000a20c0 0xc000092420] scope 0xc000010400 {
        func main
}
 [0xc000092210 0xc000092240 0xc000092270 0xc0000922a0] [token parser nil nil panic ast ast bool fmt true] []}
main
<nil>
&{<nil> 15 import 22 [0xc000092210 0xc000092240 0xc000092270 0xc0000922a0] 78}
&{<nil> <nil> 0xc0000aa020 <nil> 0}
&{28 STRING "fmt"}
parser.ParseFileast.Inspect

通过生成AST,我们可以对源代码的语法结构进行分析和操作。例如,可以通过遍历AST来寻找特定的节点,从而实现代码的查找、替换和重构等功能。

示例

解析结构体

  1. 创建名为src.go的文件,内容如下。稍后就来解析内容中的结构体
// package ast
package ast /* the name is ast */
import "fmt"

// StcA结构体说明
// descrpiton StcA
type StcA struct {
	FA string `json:"fa" form:"fa"` //测试字段A
	Fb string `json:"fb" form:"fb"`
}

type StcB struct {
	A, B, C int // associated with a, b, c
	// associated with x, y
	x, y float64    // float values
	z    complex128 // complex value
}
  1. 创建测试文件main.go
package main

import (
	"fmt"
	"go/ast"
	"go/parser"
	"go/token"
	"io/ioutil"
	"strings"
	"testing"
)

func main() {
	src, _ := ioutil.ReadFile("src.go")
	f := token.NewFileSet()
	p, err := parser.ParseFile(f, "src.go", src, parser.ParseComments)
	if err != nil {
		t.Fatal(err)
	}

	for _, v := range p.Decls {
		var (
			structComment string
			structName    string
			fieldNumber   int
		)
		if stc, ok := v.(*ast.GenDecl); ok && stc.Tok == token.TYPE {
			//fmt.Println(stc.Tok) //import、type、struct...
			if stc.Doc != nil {
				structComment = strings.TrimRight(stc.Doc.Text(), "\n")
			}
			for _, spec := range stc.Specs {
				if tp, ok := spec.(*ast.TypeSpec); ok {
					structName = tp.Name.Name
					fmt.Println("结构体名称:", structName)
					fmt.Println("结构体注释:", structComment)
					if stp, ok := tp.Type.(*ast.StructType); ok {
						if !stp.Struct.IsValid() {
							continue
						}
						fieldNumber = stp.Fields.NumFields()
						fmt.Println("字段数:", fieldNumber)
						for _, field := range stp.Fields.List {
							var (
								fieldName    string
								fieldType    string
								fieldTag     string
								fieldTagKind string
								fieldComment string
							)
							//获取字段名
							if len(field.Names) == 1 {
								fieldName = field.Names[0].Name //等同于field.Names[0].String()) //获取名称方法2
							} else if len(field.Names) > 1 {
								for _, name := range field.Names {
									fieldName = fieldName + name.String() + ","
								}
							}
							if field.Tag != nil {
								fieldTag = field.Tag.Value
								fieldTagKind = field.Tag.Kind.String()
							}
							if field.Comment != nil {
								fieldComment = strings.TrimRight(field.Comment.Text(), "\n")
							}
							if ft, ok := field.Type.(*ast.Ident); ok {
								fieldType = ft.Name
							}
							fmt.Println("\t字段名:", fieldName)
							fmt.Println("\t字段类型:", fieldType)
							fmt.Println("\t标签:", fieldTag, "标签类型", fieldTagKind)
							fmt.Println("\t字段注释:", fieldComment)
							fmt.Println("\t----------")
						}
					}
				}
			}
		}
	}
}

输出:

结构体名称: StcA
结构体注释: StcA结构体说明
descrpiton StcA
字段数: 2
	字段名: FA
	字段类型: string
	标签: `json:"fa" form:"fa"` 标签类型 STRING
	字段注释: 测试字段A
	----------
	字段名: Fb
	字段类型: string
	标签: `json:"fb" form:"fb"` 标签类型 STRING
	字段注释: 
	----------
结构体名称: StcB
结构体注释: 
字段数: 6
	字段名: A,B,C,
	字段类型: int
	标签:  标签类型 
	字段注释: associated with a, b, c
	----------
	字段名: x,y,
	字段类型: float64
	标签:  标签类型 
	字段注释: float values
	----------
	字段名: z
	字段类型: complex128
	标签:  标签类型 
	字段注释: complex value
	----------