xgo 原理探索

Go 单测 mock 方案

Mock 方法原理依赖优点缺点
接口 Mock为依赖项定义接口,并提供接口的 Mock 实现。需要定义接口和 Mock 实现。灵活,遵循 Go 的类型系统;易于替换实现。需要更多的样板代码来定义接口和 Mock 实现。
Monkey Patching(bouk/moneky)直接修改函数指针的内存地址来实现对函数的替换。内存保护;汇编代码。强大,可以 Mock 任何函数,甚至第三方库的函数。复杂,容易出错;线程不安全;依赖系统指令集。

bouk/monkey 弊端

bouk/monkey 🐒

monkey 的核心功能是能够在运行时替换某个函数的实现。

原理:

  1. 函数指针替换:在 Go 语言中,函数的地址存储在内存中。bouk/monkey 通过直接修改函数指针的内存地址来实现对函数的替换。
  2. 汇编代码:使用了汇编代码来实现对函数入口的跳转。这些汇编代码会在函数被调用时,将执行流重定向到新的函数实现。
  3. 内存保护:为了修改内存中的函数指针,bouk/monkey 需要临时修改内存页面的保护属性(例如,将页面设为可写)。在修改完毕后,它会恢复原来的保护属性。
  4. 反射与 unsafe 包:利用 Go 的反射机制和 unsafe 包,bouk/monkey 可以获取并操作函数的底层实现细节。

实现步骤:

  1. 保存原函数:在替换函数之前,bouk/monkey 会保存原始函数的指针,以便在需要时恢复或调用原始函数。
  2. 生成跳转代码:bouk/monkey 生成一段汇编跳转代码,这段代码会在函数调用时,将执行流跳转到新的函数实现。
  3. 修改函数指针:使用 unsafe 包,bouk/monkey 修改目标函数的入口地址,指向生成的跳转代码。
  4. 恢复内存保护:在完成上述修改后,恢复内存页面的保护属性。

有以下几个弊端:

  1. 如果启用了内联,Monkey 有时无法修补函数。尝试在禁用内联的情况下运行测试,例如: go test -gcflags=-l。同样的命令行参数也可以用于构建。
  2. Monkey 不能在一些面向安全的操作系统上工作,这些操作系统不允许同时写入和执行内存页。目前的方法并没有真正可靠的解决方案。
  3. 线程不安全的。
  4. 依赖指令集。

先看 xgo 怎么用

xgo 😈

代码结构如下:

.
├── greet.go
└── greet_test.go

现在在 greet.go 中有一个函数 greet

func greet(s string) string {
	return "hello " + s
}

在真实的生产环境中,greet 可能要复杂得多,它可能会依赖各种第三方 API,也可能会依赖数据库等多种外部组件。所以在测试的时候,我们希望对其进行 mock,使其返回一个固定的值,便于我们撰写单元测试。

xgo 参考了 go-monkey 的思想,但是不从 修改指令 这个途径入手,而是另辟蹊径,从 代码重写 的角度实现了 mock 的能力。

为了使用 xgo,我们需要先安装 xgo 这个命令:

go install github.com/xhd2015/xgo/cmd/xgo@latest

同时在我们的项目中需要引入 xgo 依赖:

go get "github.com/xhd2015/xgo/runtime/mock"

我们编写的 greet_test.go 如下:

package xgo_use

import (
	"testing"

	"github.com/xhd2015/xgo/runtime/mock"
)

func TestOriginGreet(t *testing.T) {
	res := greet("world")
	if res != "hello world" {
		t.Fatalf("greet() = %q; want %q", res, "hello world")
	}
}

func TestMockGreet(t *testing.T) {
	mock.Patch(greet, func(s string) string {
		return "mock " + s
	})
	res := greet("world")
	if res != "mock world" {
		t.Fatalf("greet() = %q; want %q", res, "mock world")
	}
}

可以看到在 TestMockGreet 这个单元测试中,我们将 greet 进行了 mock,返回 "mock " + s

mock.Patch(greet, func(s string) string {
  return "mock " + s
})

为了使用 xgo 的能力,我们在执行单元测试的时候,需要运行以下命令:

xgo test -v ./

输出大致如下:

➜  xgo-use git:(master) xgo test -v ./
xgo is taking a while to setup, please wait...
=== RUN   TestOriginGreet
--- PASS: TestOriginGreet (0.00s)
=== RUN   TestMockGreet
--- PASS: TestMockGreet (0.00s)
PASS
ok      xgo-explore/xgo-use     (cached)

xgo 的核心原理

xgo 的核心原理是利用 go build -toolexec 的能力。

运行以下命令:

go help build

找到 toolexec 的相关说明:

-toolexec 'cmd args'
        a program to use to invoke toolchain programs like vet and asm.
        For example, instead of running asm, the go command will run
        'cmd args /path/to/asm <arguments for asm>'.
        The TOOLEXEC_IMPORTPATH environment variable will be set,
        matching 'go list -f {{.ImportPath}}' for the package being built.

一言以蔽之:-toolexec 允许对 go 工具链进行拦截,包括 vetasmcompilelink

这种技术也被称为:插桩(stubbing)、增强(instrumentation)和代码重写(rewriting)。

-toolexec 示意图(来源:https://blog.xhd2015.xyz/zh/posts/xgo-monkey-patching-in-go-using-toolexec/)

基于上述分析,xgo 提出了 代码重写 的思路,实现了 在编译过程中插入拦截器代码 的功能:

xgo 在 go build 中的作用位置(来源:https://blog.xhd2015.xyz/zh/posts/xgo-monkey-patching-in-go-using-toolexec/)

所以上述我们的 greet.go 文件中的源代码:

func greet(s string) string {
	return "hello " + s
}

经过 xgo 编译后最终实际编译的代码如下:

import "runtime"

func greet(s string) (r0 string) {
  stop, post := runtime.__xgo_trap(Greet, &s, &r0)
  if stop {
    return
  }
  defer post()
  return "hello" + s
}

greet 函数重写变化示意图(来源:https://blog.xhd2015.xyz/zh/posts/xgo-monkey-patching-in-go-using-toolexec/)

如图所示,一旦函数被调用,它的控制流首先转移到 Trap,然后一系列拦截器将根据其目的检查当前调用是否应该被 Mock、修改、记录或停止。

如果 greet 注册了 mock 函数,那么就会在 __xgo_trap 中调用 mock 的函数,并将返回值设置到 r0 上进行返回,而跳过原始的执行逻辑。

第 1 步:死代码实现

➜  01-deadcode git:(master) tree
.
├── greet.go
├── greet_test.go
└── mock.go

我们先从最简单的实现开始,采用侵入性代码实现 xgo 的核心功能,这里我们还用不到 -toolexec

代码结构如上所示,在 mock.go 中,我们有如下代码:

var mockFuncs = sync.Map{}

func RegisterMockFunc(funcName string, fun interface{}) {
	mockFuncs.Store(funcName, fun)
}
  • mockFuncs: 用于承载函数与 mock 函数的对应关系,其中 key 为函数名称,value 为 mock 函数。我们使用 sync.Map 来保证并发安全。
  • RegisterMockFunc 用于为指定的 funcName 注册 mock 函数。

greet.go 中,我们有一个 Greet 函数:

func Greet(s string) string {
	return "hello " + s
}

如果我们要对其支持 mock,那么需要修改其实现为:

func Greet(s string) string {
	fun, ok := mockFuncs.Load("Greet")
	if ok {
		f, ok := fun.(func(s string) string)
		if ok {
			return f(s)
		}
	}
	return "hello " + s
}

在修改后的代码中,我们先判断是否存在 mock 函数,如果存在,则执行 mock 函数,否则执行原始逻辑。

现在我们在 greet_test.go 中编写测试代码:

func TestMockGreet(t *testing.T) {
	RegisterMockFunc("Greet", func(s string) string {
		return "mock " + s
	})
	res := Greet("world")
	if res != "mock world" {
		t.Fatalf("Greet() = %q; want %q", res, "mock world")
	}
}

func TestOriginGreet(t *testing.T) {
	res := Greet("world")
	if res != "hello world" {
		t.Fatalf("Greet() = %q; want %q", res, "hello world")
	}
}

执行测试:

# 单独执行 TestMockGreet
➜  01-deadcode git:(master) ✗ go test -v -run TestMockGreet
=== RUN   TestMockGreet
--- PASS: TestMockGreet (0.00s)
PASS
ok      xgo-explore/01-deadcode 0.103s

# 单独执行 TestOriginGreet
➜  01-deadcode git:(master) ✗ go test -v -run TestOriginGreet
=== RUN   TestOriginGreet
--- PASS: TestOriginGreet (0.00s)
PASS
ok      xgo-explore/01-deadcode 0.102s

# 一起执行
➜  01-deadcode git:(master) ✗ go test -v -run $Test$
=== RUN   TestMockGreet
--- PASS: TestMockGreet (0.00s)
=== RUN   TestOriginGreet
    greet_test.go:20: Greet() = "mock world"; want "hello world"
--- FAIL: TestOriginGreet (0.00s)
FAIL
exit status 1
FAIL    xgo-explore/01-deadcode 0.102s

我们会发现单独执行都是 ok 的,不过一起执行的话 TestOriginGreet 就失败了,这是因为先执行了 TestMockGreet,这个时候已经往 mockFunc 中注册了 mock 函数了,所以 TessOriginGreet 就执行失败了。

这里需要在协程层面上做 mock 隔离,xgo 的思路是在编译时注入 getg() 函数来获取当前协程信息从而实现在注册 mock 函数时进行协程隔离。本文将聚焦在 xgo 的核心原理 代码重写 上,故暂时不考虑这一块。

Ok,那么短短几行代码,我们就将 xgo 的最核心思想给展示出来了。可以看到,xgo 的核心思想是往源代码中加入 合法的 Go 代码,所以不涉及指令重写,故而只要你的机器能执行 Go 程序,天然就支持 mock 功能,这就天然达到了架构无关的兼容性了。同时我们也使用了 sync.Map 来保证了并发安全。

第 2 步:死代码拦截器

➜  02-deadcode-interceptor git:(master) tree
.
├── greet.go
├── greet_test.go
└── mock.go

在第 1 步中,这段代码我觉得有点冗长了:

fun, ok := mockFuncs.Load("Greet")
if ok {
  f, ok := fun.(func(s string) string)
  if ok {
    return f(s)
  }
}

参考 xgo 的函数签名,我们对其进行优化,在 mock.go 中加入一个 丐版拦截器

// mock.go
func InterceptMock(funcName string, arg string, result *string) bool {
	fn, ok := mockFuncs.Load(funcName)
	if ok {
		f, ok := fn.(func(s string) string)
		if ok {
			*result = f(arg)
			return true
		}
	}
	return false
}

对应 greet.goGreet 函数就修改为:

func Greet(s string) (res string) {
	if InterceptMock("Greet", s, &res) {
		return res
	}
	return "hello " + s
}

这看起来就清爽多了。再次执行测试代码,一样是可以通过的。

➜  02-deadcode-interceptor git:(master) go test -v -run TestOriginGreet
=== RUN   TestOriginGreet
--- PASS: TestOriginGreet (0.00s)
PASS
ok      xgo-explore/02-deadcode-interceptor     0.331s

➜  02-deadcode-interceptor git:(master) go test -v -run TestMockGreet
=== RUN   TestMockGreet
--- PASS: TestMockGreet (0.00s)
PASS
ok      xgo-explore/02-deadcode-interceptor     0.103s

第 3 步:toolexec 初探

➜  03-toolexec-static git:(master) tree
.
├── cmd
│   └── mytool
│       └── mytool.go
├── greet.go
├── main.go
├── mock.go
└── script.sh

这里 mock.go 没有任何变化。我们期望使用 -toolexec 来修改源代码,以实现 mock 无源代码侵入的特性,所以我们在 greet.to 中将 Greet 函数恢复为只关注实际功能的样子:

func Greet(s string) (res string) {
	return "hello " + s
}

同时为了更好地测试使用 -toolexec 编译后的运行结果,这里将 greet_test.go 删除了并新增了 main.go 文件,内容如下:

func main() {
	res := Greet("world")
	if res != "hello world" {
		log.Fatalf("Greet() = %q; want %q", res, "hello world")
	}

	RegisterMockFunc("Greet", func(s string) string {
		return "mock " + s
	})
	res = Greet("world")
	if res != "mock world" {
		log.Fatalf("Greet() = %q; want %q", res, "mock world")
	}

	log.Println("run successfully")
}

那么 -toolexec 要执行的命令怎么实现呢?在 Google 搜索 go toolexec 你会看到官方给出的一个案例:toolexec.txt。

核心部分在最下面,参考这个示例,我们来实现自己的 toolexec

mkdir -p cmd/mytool
touch cmd/mytool/mytool.go

mytool.go 中,我们先写这么点代码,看一下会输出什么。

func main() {
	tool, args := os.Args[1], os.Args[2:]
	if len(args) > 0 && args[0] == "-V=full" {
		// don't do anything to infuence the version full output.
	} else if len(args) > 0 {
		fmt.Printf("tool: %s\n", tool)
		fmt.Printf("args: %v\n", args)
	}
  // 继续执行之前的命令
	cmd := exec.Command(tool, args...)
	cmd.Stdout = os.Stdout
	cmd.Stderr = os.Stderr

	if err := cmd.Run(); err != nil {
		log.Fatalf("run command error: %v\n", err)
	}
}

这里我们企图输出执行的工具 tool 及传给它的参数 args。由于 -V=full 的作用是在终端输出版本信息,所以我们要跳过它,避免产生干扰。输出日志后,我们暂且先继续执行原始的命令,不对编译过程做其他的干扰。

Ok,现在就来看看这个 -toolexec 到底做了什么,在 03-toolexec-static 目录下执行以下命令:

# 清除缓存,一直使用最新的编译结果
go clean -cache -modcache -i -r
# 编译 mytool
go build ./cmd/mytool
# 编译业务程序
go build -toolexec=./mytool -o main

因为这几个命令经常会用到,所以我们可以将其封装到 script.sh 文件中:

touch script.sh
chmod +x script.sh

内容如下:

#!/bin/bash

go clean -cache -modcache -i -r
go build ./cmd/mytool
go build -toolexec=./mytool -o main

执行上述命令后,可以看到以下输出:

➜  03-toolexec-static git:(master) ./script.sh
# xgo-explore/03-toolexec-static
tool: /opt/homebrew/Cellar/go/1.22.3/libexec/pkg/tool/darwin_arm64/compile
args: [-o $WORK/b001/_pkg_.a -trimpath $WORK/b001=> -p main -lang=go1.22 -complete -buildid PcS9clqF_ny_Ds5N0i_s/PcS9clqF_ny_Ds5N0i_s -goversion go1.22.3 -c=4 -shared -nolocalimports -importcfg $WORK/b001/importcfg -pack ./greet.go ./main.go ./mock.go]
# xgo-explore/03-toolexec-static
tool: /opt/homebrew/Cellar/go/1.22.3/libexec/pkg/tool/darwin_arm64/link
args: [-o $WORK/b001/exe/a.out -importcfg $WORK/b001/importcfg.link -buildmode=pie -buildid=KgnnCoU_6enHkOm-T62Z/PcS9clqF_ny_Ds5N0i_s/H80dtgGZw1L8mTtVqJBf/KgnnCoU_6enHkOm-T62Z -extld=cc $WORK/b001/_pkg_.a]

可以看到执行了 compilelink 两个工具,compile 是编译过程,将生成 {}.out 文件,而 link 是将多个 {}.out 文件链接成一个可执行文件。这是很经典的编译过程,如果对 Go 语言的编译过程感兴趣,也可以参考官方的 Go Compile Readme,或者笔者撰写的 Go1.21.0 程序编译过程。

这里我们需要重点关注的是 compile 命令,它是负责编译源代码的,涉及到的源代码文件会通过 -pack ./greet.go ./main.go ./mock.go 传递给 compile 命令。

结合 -toolexec 的帮助信息:

-toolexec 'cmd args'
        a program to use to invoke toolchain programs like vet and asm.
        For example, instead of running asm, the go command will run
        'cmd args /path/to/asm <arguments for asm>'.
        The TOOLEXEC_IMPORTPATH environment variable will be set,
        matching 'go list -f {{.ImportPath}}' for the package being built.

我们只需要在执行 compile 命令之前,在 cmd args 这个环节,进行 代码重写 就可以实现我们想要的功能了。

我们现在是要对 greet.go 里面的 Greet 函数进行重写,先看看之前的代码:

package main

func Greet(s string) (res string) {
	return "hello " + s
}

重写后的代码应该跟我们之前 第 2 步 是一样的:

package main

func Greet(s string) (res string) {
	if InterceptMock("Greet", s, &res) {
	 	return res
    }
	return "hello " + s
}

这里有 n 多种方式可以做到,现在笔者决定使用最暴力的方式,直接临时创建一个包含这段代码的文件 tmp.go,并替换掉传给 compile 的参数,即将 -pack ./greet.go ./main.go ./mock.go 替换为 -pack tmp.go ./main.go ./mock.go

综上,cmd/mytool/mytool/go 实现的代码如下:

func main() {
	tool, args := os.Args[1], os.Args[2:]

	if len(args) > 0 && args[0] == "-V=full" {
		// don't do anything to infuence the version full output.
	} else if len(args) > 0 {
		if filepath.Base(tool) == "compile" {
			index := findGreetFile(args)
			if index > -1 {
				f, err := os.Create("tmp.go")
				if err != nil {
					log.Fatalf("create tmp.go error: %v\n", err)
				}
				defer f.Close()
				defer os.Remove("tmp.go")
				_, _ = f.WriteString(newCode)
				args[index] = "tmp.go"
			}
		}
		fmt.Printf("tool: %s\n", tool)
		fmt.Printf("args: %v\n", args)
	}
	// 继续执行之前的命令
	cmd := exec.Command(tool, args...)
	cmd.Stdout = os.Stdout
	cmd.Stderr = os.Stderr

	if err := cmd.Run(); err != nil {
		log.Fatalf("run command error: %v\n", err)
	}
}

func findGreetFile(args []string) int {
	for i, arg := range args {
		if strings.Contains(arg, "greet.go") {
			return i
		}
	}
	return -1
}

var newCode = `
package main

func Greet(s string) (res string) {
	if InterceptMock("Greet", s, &res) {
	 	return res
    }
	return "hello " + s
}
`

这里我先使用 findGreetFile 来查找 greet.go 文件所处的参数位置,如果找到了,则生成新的 tmp.go 文件,并替换参数,最后在 本次 compile 命令执行完毕后,删除 tmp.go,“毁尸灭迹”。

执行 ./script.sh 重新编译:

➜  03-toolexec-static git:(master) ✗ ./script.sh
# xgo-explore/03-toolexec-static
tool: /opt/homebrew/Cellar/go/1.22.3/libexec/pkg/tool/darwin_arm64/compile
args: [-o $WORK/b001/_pkg_.a -trimpath $WORK/b001=> -p main -lang=go1.22 -complete -buildid PcS9clqF_ny_Ds5N0i_s/PcS9clqF_ny_Ds5N0i_s -goversion go1.22.3 -c=4 -shared -nolocalimports -importcfg $WORK/b001/importcfg -pack tmp.go ./main.go ./mock.go]
# xgo-explore/03-toolexec-static
tool: /opt/homebrew/Cellar/go/1.22.3/libexec/pkg/tool/darwin_arm64/link
args: [-o $WORK/b001/exe/a.out -importcfg $WORK/b001/importcfg.link -buildmode=pie -buildid=KgnnCoU_6enHkOm-T62Z/PcS9clqF_ny_Ds5N0i_s/H80dtgGZw1L8mTtVqJBf/KgnnCoU_6enHkOm-T62Z -extld=cc $WORK/b001/_pkg_.a]

输出的结果中可以看到已经将 compile 的参数替换为 -pack tmp.go ./main.go ./mock.go 了。

现在我们来执行生成的程序文件,可以看到是执行成功的。

➜  03-toolexec-static git:(master) ✗ ./main
2024/05/23 17:53:52 run successfully

如果我们不使用 -toolexec,是执行不成功的:

➜  03-toolexec-static git:(master) ✗ go clean -cache -modcache -i -r
➜  03-toolexec-static git:(master) ✗ go build -o main
➜  03-toolexec-static git:(master) ✗ ./main
2024/05/23 17:54:33 Greet() = "hello world"; want "mock world"

第 4 步:使用 AST 在函数前插入代码

➜  04-toolexec-ast git:(master) ✗ tree
.
├── cmd
│   └── mytool
│       └── mytool.go
├── greet.go
├── main.go
├── mock.go
└── script.sh

暴力替换源代码文件的方式可能是不太优雅哈,假如我们的 greet.go 内容改成下面这样:

package main

func Greet(s string) (res string) {
	return "hello " + s
}

func Greet2(s string) (res string) {
	return "hello 2 " + s
}

如果我们想对 Greet2 也进行 代码重写,那就需要修改前面 newCode 字段的内容,而且它是写死的,确实不太优雅。现在我们正式来面对这件事,对比修改后的函数:

func Greet(s string) (res string) {
	if InterceptMock("Greet", s, &res) {
	 	return res
    }
	return "hello " + s
}

其实就是在每个函数前加上这么一段:

if InterceptMock("Greet", s, &res) {
	return res
}

了解过编译原理的读者应该可以想到,我们可以通过操作源代码的 AST 结构,往函数的开头插入这段代码即可。如果我们先不考虑参数和返回值的话,那这段代码我们需要替换的地方就是函数名称了,所以它的结构如下:

if InterceptMock("${funcName}", s, &res) {
	return res
}

这里我们需要用到几个标准库工具:

  • go/ast: 包定义了 Go 编程语言的抽象语法树(AST),核心有以下几种类型:
    • File: 表示一个 Go 源文件。
    • Decl: 表示一个声明,包括函数声明、变量声明、类型声明等。
    • Stmt: 表示一个语句。
    • Expr: 表示一个表达式。
  • go/token: 定义了处理 Go 源代码的词法元素的基础设施,包括位置、标记和标识符等。这个包提供了用于管理源代码位置的信息,可以帮助定位代码中的特定部分。
  • go/parser: 将一个 .go 文件以解析成 AST 结构。
  • go/printer: 提供了将 AST 格式化并输出为 Go 源码的功能

修改后的 cmd/mytool/mytool.go 代码如下:

func main() {
	tool, args := os.Args[1], os.Args[2:]

	if len(args) > 0 && args[0] == "-V=full" {
		// don't do anything to infuence the version full output.
	} else if len(args) > 0 {
		if filepath.Base(tool) == "compile" {
			index := findGreetFile(args)
			if index > -1 {
				filename := args[index]
				f, err := os.Create("tmp.go")
				defer f.Close()
				defer os.Remove("tmp.go")
				if err != nil {
					log.Fatalf("create tmp.go error: %v\n", err)
				}
				_, _ = f.WriteString(insertCode(filename))
				args[index] = "tmp.go"
			}
		}
		fmt.Printf("tool: %s\n", tool)
		fmt.Printf("args: %v\n", args)
	}
	// 继续执行之前的命令
	cmd := exec.Command(tool, args...)
	cmd.Stdout = os.Stdout
	cmd.Stderr = os.Stderr

	if err := cmd.Run(); err != nil {
		log.Fatalf("run command error: %v\n", err)
	}
}

func findGreetFile(args []string) int {
	for i, arg := range args {
		if strings.Contains(arg, "greet.go") {
			return i
		}
	}
	return -1
}

func insertCode(filename string) string {
	fset := token.NewFileSet()
	fast, err := parser.ParseFile(fset, filename, nil, parser.AllErrors)
	if err != nil {
		log.Fatalf("parse file error: %v\n", err)
	}

	for _, decl := range fast.Decls {
		fun, ok := decl.(*ast.FuncDecl)
		if !ok {
			continue
		}

		f, err := os.Create("tmp2.go")
		if err != nil {
			log.Fatalf("create tmp2.go error: %v\n", err)
		}
		_, _ = f.WriteString(fmt.Sprintf(newCodeFormat, fun.Name.Name))
		f.Close()

		tmpFset := token.NewFileSet()
		tmpF, err := parser.ParseFile(tmpFset, "tmp2.go", nil, parser.AllErrors)
		if err != nil {
			log.Fatalf("parse tmp2.go error: %v\n", err)
		}
		fun.Body.List = append(tmpF.Decls[0].(*ast.FuncDecl).Body.List, fun.Body.List...)
		os.Remove("tmp2.go")
	}

	var buf bytes.Buffer
	printer.Fprint(&buf, fset, fast)

	fmt.Println(buf.String())

	return buf.String()
}

var newCodeFormat = `
package main

func TmpFunc() {
	if InterceptMock("%s", s, &res) {
	 	return res
    }
}
`

核心的修改在于 insertCode 函数:

  1. 使用 parser.ParseFile 将源代码文件解析成 AST 结构;

  2. 遍历 AST 结构,找到所有的声明(Decl)结构,并使用 decl(.ast.FuncDecl) 找到所有的函数;

    FuncDecl struct {
      Doc  *CommentGroup // associated documentation; or nil
      Recv *FieldList    // receiver (methods); or nil (functions)
      Name *Ident        // function/method name
      Type *FuncType     // function signature: type and value parameters, results, and position of "func" keyword
      Body *BlockStmt    // function body; or nil for external (non-Go) function
    }
    
    BlockStmt struct {
      Lbrace token.Pos // position of "{"
      List   []Stmt
      Rbrace token.Pos // position of "}", if any (may be absent due to syntax error)
    }
    
  3. 查看 ast.FuncDecl 的结构后,可以得出下一步就是往 FuncDecl.Body.List 列表前面插入一些 Stmt

  4. 笔者没找到类似 parseStmt 方法,所以取了个巧,我定义了一段代码的 format,里面的 %s 会使用 fun.Name.Name 获取函数名并进行替换。

    var newCodeFormat = `
    package main
    
    func TmpFunc() {
    	if InterceptMock("%s", s, &res) {
    	 	return res
        }
    }
    `
    
  5. 创建一个临时文件 tmp2.go 并写入格式化后的代码,然后再次调用 parser.ParseFile 得到解析这段代码的抽象语法树结构 tmpF 了;

  6. 然后通过 tmpF.Decls[0].(*ast.FuncDecl).Body.List 就可以得到 TmpFunc 中的语句 Stmt 了;

  7. 将其加在源代码函数的前面即可:fun.Body.List = append(tmpF.Decls[0].(*ast.FuncDecl).Body.List, fun.Body.List...)

  8. 然后再使用 go/printer 将修改后的 AST 输出为新文件内容。

通过上述步骤,我们就可以为 greet.go 中的每个函数前面都插入打桩代码了。

修改 main.go 里面的内容,加入对 Greet2 的测试:

func main() {
	res := Greet("world")
	if res != "hello world" {
		log.Fatalf("Greet() = %q; want %q", res, "hello world")
	}

	RegisterMockFunc("Greet", func(s string) string {
		return "mock " + s
	})
	res = Greet("world")
	if res != "mock world" {
		log.Fatalf("Greet() = %q; want %q", res, "mock world")
	}

	log.Println("run greet 1 successfully")

	RegisterMockFunc("Greet2", func(s string) string {
		return "mock 2 " + s
	})
	res = Greet2("world")
	if res != "mock 2 world" {
		log.Fatalf("Greet2() = %q; want %q", res, "mock 2 world")
	}

	log.Println("run greet 2 successfully")
}

执行脚本:

./script.sh

输出应该还是跟之前是一样的,我们运行生成的可执行函数,得到如下结果那就说明我们又成功进了一步了~

➜  04-toolexec-ast git:(master) ✗ ./main
2024/05/23 20:03:22 run greet 1 successfully
2024/05/23 20:03:22 run greet 2 successfully

第 5 步:使用 reflect 反射动态获取参数和返回值名称

➜  05-toolexec-general git:(master) ✗ tree
.
├── cmd
│   └── mytool
│       └── mytool.go
├── greet.go
├── main.go
├── mock.go
└── script.sh

接下来我们来处理函数签名中的参数和返回值部分,我们的样板代码中,写死了参数的名称和返回值的名称,现在我们需要来动态获取函数参数的名称和返回值的名称,如果返回值没有名称,那我们还需要手动设置名称。

我们将 greet.to 修改为以下内容:

func Greet(s string) (res string) {
	return "hello " + s
}

func Greet2(s2 string) (res2 string) {
	return "hello 2 " + s2
}

func Greet3(s3 string) string {
	return "hello 3 " + s3
}

函数的信息当然都在前面获得的 ast.FuncDecl 结构中,再次观察其结构:

FuncDecl struct {
		Doc  *CommentGroup // associated documentation; or nil
		Recv *FieldList    // receiver (methods); or nil (functions)
		Name *Ident        // function/method name
		Type *FuncType     // function signature: type and value parameters, results, and position of "func" keyword
		Body *BlockStmt    // function body; or nil for external (non-Go) function
	}

通过注释就可以知道 Type 字段就包含了参数和返回值的相关信息,查看 FuncType 结构,如下:

FuncType struct {
  Func       token.Pos  // position of "func" keyword (token.NoPos if there is no "func")
  TypeParams *FieldList // type parameters; or nil
  Params     *FieldList // (incoming) parameters; non-nil
  Results    *FieldList // (outgoing) results; or nil
}
  • Params:函数参数
  • Results:函数返回值

查看 FieldList 结构,可知参数列表和返回值列表都在相应的 List 字段中,而其中的 Names 字段就是参数的名称了。

type FieldList struct {
	Opening token.Pos // position of opening parenthesis/brace/bracket, if any
	List    []*Field  // field list; or nil
	Closing token.Pos // position of closing parenthesis/brace/bracket, if any
}

type Field struct {
	Doc     *CommentGroup // associated documentation; or nil
	Names   []*Ident      // field/method/(type) parameter names; or nil
	Type    Expr          // field/method/parameter type; or nil
	Tag     *BasicLit     // field tag; or nil
	Comment *CommentGroup // line comments; or nil
}

补充一下,这里为什么 Names 类型是 []*Ident 呢?因为函数有以下的命名方式:

func hello(s1, s2 string) (r1, r1 string) {}

那么在当下,只有 1 个参数和只有 1 个返回值的情况下,我们就可以通过 fun.Type.Params.List[0].Names[0].Name 来获取参数名称,也可以通过 fun.Type.Results.List[0].Names 来获取返回值名称,如果返回值没有名称,那我们就为其设置名称 __xgo_res_1 并写回源 AST 结构。这样就都有名称,就很好处理了。

经上分析, cmd/mytool/mytool.go 中我们只需要修改 insertCode 部分,修改的结果如下:

func insertCode(filename string) string {
	fset := token.NewFileSet()
	fast, err := parser.ParseFile(fset, filename, nil, parser.AllErrors)
	if err != nil {
		log.Fatalf("parse file error: %v\n", err)
	}

	for _, decl := range fast.Decls {
		fun, ok := decl.(*ast.FuncDecl)
		if !ok {
			continue
		}

		f, err := os.Create("tmp.go")
		if err != nil {
			log.Fatalf("create tmp.go error: %v\n", err)
		}
		_, _ = f.WriteString(newCode(fun))
		f.Close()

		tmpFset := token.NewFileSet()
		tmpF, err := parser.ParseFile(tmpFset, "tmp.go", nil, parser.AllErrors)
		if err != nil {
			log.Fatalf("parse tmp.go error: %v\n", err)
		}
		fun.Body.List = append(tmpF.Decls[0].(*ast.FuncDecl).Body.List, fun.Body.List...)
		os.Remove("tmp.go")
	}

	var buf bytes.Buffer
	printer.Fprint(&buf, fset, fast)

	fmt.Println(buf.String())

	return buf.String()
}

func newCode(fun *ast.FuncDecl) string {

	/*
		&{Doc:<nil> Names:[s] Type:string Tag:<nil> Comment:<nil>}
		&{Doc:<nil> Names:[res] Type:string Tag:<nil> Comment:<nil>}
		&{Doc:<nil> Names:[s2] Type:string Tag:<nil> Comment:<nil>}
		&{Doc:<nil> Names:[res2] Type:string Tag:<nil> Comment:<nil>}
		&{Doc:<nil> Names:[s3] Type:string Tag:<nil> Comment:<nil>}
		&{Doc:<nil> Names:[] Type:string Tag:<nil> Comment:<nil>}
	*/

	// 函数名称
	funcName := fun.Name.Name

	// 参数列表
	argName := fun.Type.Params.List[0].Names[0].Name

	// 返回值列表
	resNames := fun.Type.Results.List[0].Names
	if len(resNames) == 0 {
		resNames = append(resNames, &ast.Ident{Name: "_xgo_res_1"})
		fun.Type.Results.List[0].Names = resNames
	}
	resName := resNames[0].Name
	return fmt.Sprintf(newCodeFormat, funcName, argName, resName, resName)
}

var newCodeFormat = `
package main

func TmpFunc() {
	if InterceptMock("%s", %s, &%s) {
	 	return %s
    }
}
`

现在我们就可以动态获取参数名称和返回值名称了。

修改我们的 main.go,以测试所有的情况:

func main() {
	res := Greet("world")
	if res != "hello world" {
		log.Fatalf("Greet() = %q; want %q", res, "hello world")
	}

	RegisterMockFunc("Greet", func(s string) string {
		return "mock " + s
	})
	res = Greet("world")
	if res != "mock world" {
		log.Fatalf("Greet() = %q; want %q", res, "mock world")
	}

	log.Println("run greet 1 successfully")

	RegisterMockFunc("Greet2", func(s string) string {
		return "mock 2 " + s
	})
	res = Greet2("world")
	if res != "mock 2 world" {
		log.Fatalf("Greet2() = %q; want %q", res, "mock 2 world")
	}

	log.Println("run greet 2 successfully")

	RegisterMockFunc("Greet3", func(s string) string {
		return "mock 3 " + s
	})
	res = Greet3("world")
	if res != "mock 3 world" {
		log.Fatalf("Greet3() = %q; want %q", res, "mock 3 world")
	}

	log.Println("run greet 3 successfully")
}

执行编译脚本:

./script.sh

执行编译产生的可执行程序,输出如下就说明我们又成功进了一大步~

➜  05-toolexec-general git:(master) ✗ ./main
2024/05/23 20:15:08 run greet 1 successfully
2024/05/23 20:15:08 run greet 2 successfully
2024/05/23 20:15:08 run greet 3 successfully

第 6 步:支持多参数和多返回值

➜  06-toolexec-multi git:(master) ✗ tree
.
├── cmd
│   └── mytool
│       └── mytool.go
├── greet.go
├── main.go
├── mock.go
└── script.sh

本文的最后一步,我们来面对一下多参数和多返回值的问题。假设我们又如下函数:

func Pair1(s1, s2 string) (res string) {
	return "pair 1 " + s1 + " " + s2
}

这个时候我们 代码重写 后应该长什么样子呢?可以是下面这样的:

func Pair1(s1, s2 string) (res string) {
	if InterceptMock("Pair1", s1, s2, &res) {
		return res
	}
	return "pair 1 " + s1 + " " + s2
}

按照这个思路,下面这个函数呢?

func Pair2(s1, s2 string) (res1, res2 string) {
	return "pair 1 " + s1, "pair 2 " + s2
}

那就是这样的?

func Pair2(s1, s2 string) (res1, res2 string) {
	if InterceptMock("Pair2", s1, s2, &res1, &res2) {
		return res1, res2
	}
	return "pair 1 " + s1, "pair 2 " + s2
}

这种思路当然也能实现,换一种更优雅的思路呢?既然是一个列表,那么就可以用切片来承载,也就是可以是这样的:

func Pair2(s1, s2 string) (res1, res2 string) {
	if InterceptMock("Pair2", []interface{}{s1, s2}, []interface{}{&res1, &res2}) {
		return res1, res2
	}
	return "pair 1 " + s1, "pair 2 " + s2
}

那我们就可以抽象出插入代码的模板了:

if InterceptMock("${funcName}", []interface{}{${paramList}}, []interface{}{${returnListWith&}}) {
  return ${returnListWithout&}
}

为了实现这个,我们需要先修改一下 mock.go 中的 InterceptMock 函数:

func InterceptMock(funcName string, args []interface{}, results []interface{}) bool {
	mockFn, ok := mockFuncs.Load(funcName)
	if !ok {
		return false
	}


	in := make([]reflect.Value, len(args))
	for i, arg := range args {
		in[i] = reflect.ValueOf(arg)
	}

  mockFnValue := reflect.ValueOf(mockFn)
	out := mockFnValue.Call(in)
	if len(out) != len(results) {
		panic("mock function return value number is not equal to results number")
	}

	for i, result := range results {
		reflect.ValueOf(result).Elem().Set(out[i])
	}
	return true
}

拦截器的具体实现如下:

  1. 判断是否注册了 mock 函数,没有则直接返回;
  2. 将所有参数都放到 []refect.Value 中;
  3. 通过反射 refect.ValueOf 获取 mockFn 的值;
  4. 调用 mockFnValue.Call() 来执行函数,并返回结果列表;
  5. 遍历传进来的返回值引用列表,调用 reflect.ValueOf(result).Elem().Set(out[i]) 将返回值设置回去。

现在我们来修改我们的 -toolexec 工具,来根据函数的 AST 结构,获取参数列表和返回值列表,生成代插入的模板代码,并将其插入到每个函数的开头。这次在 cmd/mytool/mytool.go 中,我们只需修改 newCode 函数:

func insertCode(filename string) string {
	fset := token.NewFileSet()
	fast, err := parser.ParseFile(fset, filename, nil, parser.AllErrors)
	if err != nil {
		log.Fatalf("parse file error: %v\n", err)
	}

	for _, decl := range fast.Decls {
		fun, ok := decl.(*ast.FuncDecl)
		if !ok {
			continue
		}

		f, err := os.Create("tmp.go")
		if err != nil {
			log.Fatalf("create tmp.go error: %v\n", err)
		}
		_, _ = f.WriteString(newCode(fun))
		f.Close()

		tmpFset := token.NewFileSet()
		tmpF, err := parser.ParseFile(tmpFset, "tmp.go", nil, parser.AllErrors)
		if err != nil {
			log.Fatalf("parse tmp.go error: %v\n", err)
		}
		fun.Body.List = append(tmpF.Decls[0].(*ast.FuncDecl).Body.List, fun.Body.List...)
		os.Remove("tmp.go")
	}

	var buf bytes.Buffer
	printer.Fprint(&buf, fset, fast)

	fmt.Println(buf.String())

	return buf.String()
}

func newCode(fun *ast.FuncDecl) string {
	// 函数名称
	funcName := fun.Name.Name

	// 参数列表
	args := make([]string, 0)
	for _, arg := range fun.Type.Params.List {
		for _, name := range arg.Names {
			args = append(args, name.Name)
		}
	}
	// 返回值列表
	returns := make([]string, 0)
	returnRefs := make([]string, 0)
	returnNames := fun.Type.Results.List[0].Names
	if len(returnNames) == 0 {
		for i := 0; i < fun.Type.Results.NumFields(); i++ {
			fun.Type.Results.List[0].Names = append(fun.Type.Results.List[0].Names,
				&ast.Ident{Name: fmt.Sprintf("_xgo_res_%d", i+1)})
		}
	}
	for _, re := range fun.Type.Results.List[0].Names {
		returns = append(returns, re.Name)
		returnRefs = append(returnRefs, "&"+re.Name)
	}
	return fmt.Sprintf(newCodeFormat,
		funcName,
		strings.Join(args, ","),
		strings.Join(returnRefs, ","),
		strings.Join(returns, ","))
}

var newCodeFormat = `
package main

func TmpFunc() {
	if InterceptMock("%s", []interface{}{%s}, []interface{}{%s}) {
		return %s
	}
}
`

思路跟之前第 5 步大同小异,不过是用遍历的方式来支持多个参数和多个返回值罢了。

现在我们为 greet.go 添加更多的测试函数,代码如下:

func Greet(s string) (res string) {
	return "hello " + s
}

func Greet2(s2 string) (res2 string) {
	return "hello 2 " + s2
}

func Greet3(s3 string) string {
	return "hello 3 " + s3
}

func Pair1(s1, s2 string) (res string) {
	return "pair 1 " + s1 + " " + s2
}

func Pair2(s1, s2 string) (res1, res2 string) {
	return "pair 1 " + s1, "pair 2 " + s2
}

func Other(i int, s string, f float64) string {
	return fmt.Sprintf("int: %d, string: %s, float: %f", i, s, f)
}

为了测试,我们再次修改 main.go,使其覆盖所有的情况:

func main() {

	RegisterMockFunc("Other", func(i int, s string, f float64) string {
		return fmt.Sprintf("mock %d %s %.2f", i, s, f)
	})
	res := Other(1, "hello", 3.14)
	if res != "mock 1 hello 3.14" {
		log.Fatalf("Other() = %q; want %q", res, "mock 1 hello 3.14")
	}
	log.Println("run other successfully")

	RegisterMockFunc("Pair1", func(s1, s2 string) string {
		return "mock 1 " + s1 + " " + s2
	})
	res = Pair1("hello", "world")
	if res != "mock 1 hello world" {
		log.Fatalf("Pair1() = %q; want %q", res, "mock 1 hello world")
	}
	log.Println("run pair1 successfully")

	RegisterMockFunc("Pair2", func(s1, s2 string) (string, string) {
		return "mock 2 " + s1, "mock 2 " + s2
	})
	res1, res2 := Pair2("hello", "world")
	if res1 != "mock 2 hello" || res2 != "mock 2 world" {
		log.Fatalf("Pair2() = %q, %q; want %q, %q", res1, res2, "mock 2 hello", "mock 2 world")
	}
	log.Println("run pair2 successfully")

	res = Greet("world")
	if res != "hello world" {
		log.Fatalf("Greet() = %q; want %q", res, "hello world")
	}

	RegisterMockFunc("Greet", func(s string) string {
		return "mock " + s
	})
	res = Greet("world")
	if res != "mock world" {
		log.Fatalf("Greet() = %q; want %q", res, "mock world")
	}

	log.Println("run greet 1 successfully")

	RegisterMockFunc("Greet2", func(s string) string {
		return "mock 2 " + s
	})
	res = Greet2("world")
	if res != "mock 2 world" {
		log.Fatalf("Greet2() = %q; want %q", res, "mock 2 world")
	}

	log.Println("run greet 2 successfully")

	RegisterMockFunc("Greet3", func(s string) string {
		return "mock 3 " + s
	})
	res = Greet3("world")
	if res != "mock 3 world" {
		log.Fatalf("Greet3() = %q; want %q", res, "mock 3 world")
	}

	log.Println("run greet 3 successfully")
}

编译代码:

./script.sh

执行生成的可执行程序,如果有以下输出,那我们就又成功进了一大大步了~

➜  06-toolexec-multi git:(master) ✗ ./main
2024/05/23 20:31:10 run other successfully
2024/05/23 20:31:10 run pair1 successfully
2024/05/23 20:31:10 run pair2 successfully
2024/05/23 20:31:10 run greet 1 successfully
2024/05/23 20:31:10 run greet 2 successfully
2024/05/23 20:31:10 run greet 3 successfully

更进一步

通过上面 6 个简单的小阶段,我们就已经把 xgo 最最核心的功能给实现了,在一些小场景下还勉强能用?🤡

我们来看看包含测试代码和样例函数,总共用了多少代码:

➜  06-toolexec-multi git:(master) ✗ tokei .
===============================================================================
 Language            Files        Lines         Code     Comments       Blanks
===============================================================================
 Go                      4          281          224           11           46
 Shell                   1            5            3            1            1
===============================================================================
 Total                   5          286          227           12           47
===============================================================================

短短 224 行代码,这是一个非常了不起的成就!

当然,优秀的读者肯定可以发现我们这个 丐版 xgo 有太多的不足和缺陷了。这是必然的,我们来看看 xgo 截止 1.0.37 版本,总共有多少行代码:

➜  xgo git:(master) tokei .
===============================================================================
 Language            Files        Lines         Code     Comments       Blanks
===============================================================================
 BASH                    1          104           81           11           12
 CSS                     1          153          118            5           30
 Go                    369        33232        26836         2588         3808
 JavaScript              1          170          146           10           14
 JSON                    2          435          435            0            0
 PowerShell              1           28           16            3            9
 Shell                   3          288          251            4           33
 SVG                     1           41           41            0            0
 Plain Text              7          192            0          174           18
-------------------------------------------------------------------------------
 HTML                    1           19           16            3            0
 |- JavaScript           1            6            6            0            0
 (Total)                             25           22            3            0
-------------------------------------------------------------------------------
 Markdown               17         1455            0         1083          372
 |- Go                   8          820          635           72          113
 |- JSON                 1           80           80            0            0
 (Total)                           2355          715         1155          485
===============================================================================
 Total                 404        36117        27940         3881         4296
===============================================================================

光 Go 代码就有 26836 行了。所以可知 xgo 的作者是做了很多的付出和努力的。不过我们用了不到百分之一的代码量,就将 xgo 最核心的原理展示得淋漓尽致了,感兴趣的读者可以进一步阅读 xgo 的源码,可以进一步探索如何抽象出更通用更简洁更易扩展的 interceptor,如何支持协程隔离,如何优化依赖管理,以及如何实现其他的 trace、coverage 功能。再次为 xgo 打 call 👏!

参考

  • xgo repo
  • xgo: 基于代码重写实现 Monkey Patch 和 Trace
  • go compile README
  • xgo: 在 go 中使用-toolexec 实现猴子补丁

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

深度学习网络结构之---Inception

目录 一、Inception名称的由来 二、Inception结构 三、Inception v2 四、Inception v3 1、深度网络的通用设计原则 2.卷积分解&#xff08;Factorizing Convolutions&#xff09; 3.对称卷积分解 3.非对称卷积分解 五、Inception v4 一、Inception名称的由来 Inception网…

推荐一款好用的读论文软件操作方法

步骤&#xff1a; 1. 使用一译 —— 文档和论文翻译、对照阅读、讨论和社区 2.上传自己想要翻译的论文即可。 示例 Planing论文双语翻译 1.1 Parting with Misconceptions about Learning-based Vehicle Motion Planning 中英文对照阅读 1.2 Rethinking Imitation-based Pl…

3.多层感知机

目录 1.感知机训练感知机XOR问题&#xff08;Minsky&Papert 1969&#xff09; AI的第一个寒冬总结 2.多层感知机(MLP)学习XOR单隐藏层&#xff08;全连接层&#xff09;激活函数&#xff1a;Sigmoid激活函数&#xff1a;Tanh激活函数&#xff1a;ReLu 最常用的 因为计算速度…

AMSR-MODIS 边界层水汽 L3 每日 1 度 x 1 度 V1、V2 版本数据集

AMSR-MODIS Boundary Layer Water Vapor L3 Daily 1 degree x 1 degree V1 (AMDBLWV) at GES DISC AMSR-MODIS Boundary Layer Water Vapor L3 Daily 1 degree x 1 degree V2 (AMDBLWV) at GES DISC 简介 该数据集可估算均匀云层下的海洋边界层水汽。AMSR-E 和 AMSR-2 的微波…

使用libcurl实现简单的HTTP访问

代码; #include <stdio.h> #include <stdlib.h> #include <curl/curl.h> // 包含libcurl库 FILE *fp; // 定义一个文件标识符 size_t write_data(void *ptr,size_t size,size_t nmemb,void *stream) { // 定义回调函数&#xff0c;用于将…

MGRS坐标

一 概述 MGRS坐标系统&#xff0c;即军事格网参考系统&#xff0c;是北约(NATO)军事组织使用的标准坐标系统。它基于UTM&#xff08;通用横向墨卡托&#xff09;系统&#xff0c;并将每个UTM区域进一步划分为100km100km的小方块。这些方块通过两个相连的字母标识&#xff0c;其…

华为云开发者社区活动-基于MindNLP的ChatGLM-6B聊天机器人体验

MindNLP ChatGLM-6B StreamChat 本案例基于MindNLP和ChatGLM-6B实现一个聊天应用。支持流式回复。 本活动通过配置环境&#xff0c;模型接入&#xff0c;以及gradio前端界面搭建&#xff0c;实现了聊天机器人的功能。 以下是一些体验记录&#xff1a; 有兴趣的可以通过以下链…

详细解析找不到msvcp120.dll文件的原因及解决方法

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“找不到msvcp120.dll”。这个错误提示通常出现在运行某些程序或游戏时&#xff0c;给使用者带来了困扰。那么&#xff0c;究竟是什么原因导致了这个问题的出现&#xff1f;又该如何解决呢&a…

【每日刷题】Day64

【每日刷题】Day64 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. LCP 67. 装饰树 - 力扣&#xff08;LeetCode&#xff09; 3. 1315. 祖父节点值为偶数的节点和 - 力…

PyQT5 中关于 QCheckBox 的勾选状态的一点小细节

一、QCheckBox 是 PyQt5 中的一个用于创建复选框的控件&#xff0c;以下是其一些常见方法和属性&#xff1a; setChecked: 设置复选框的选中状态。isChecked: 检查复选框是否被选中。text: 设置或获取复选框的文本。state: 获取复选框的状态&#xff08;无、选中、不可用等&am…

公差基础-配合(互换性和测量基础)-2

过盈配合&#xff1a; 配合的种类&#xff1a; 三种&#xff1a;间隙&#xff0c;过渡&#xff0c;过盈配 间隙配合&#xff1a; 过盈配合&#xff1a; 过渡配合&#xff1a; 间隙量&#xff1a;最大间隙减去最小间隙&#xff1b; 配合的公差怎么算&#xff1a; 练习&#xff…

Git 代码管理规范 !

分支命名 master 分支 master 为主分支&#xff0c;也是用于部署生产环境的分支&#xff0c;需要确保master分支稳定性。master 分支一般由 release 以及 hotfix 分支合并&#xff0c;任何时间都不能直接修改代码。 develop 分支 develop 为开发环境分支&#xff0c;始终保持最…

如何在 Go 应用程序中使用检索增强生成(RAG)

本文将帮助大家实现 RAG &#xff08;使用 LangChain 和 PostgreSQL &#xff09;以提高 LLM 输出的准确性和相关性。 得益于强大的机器学习模型&#xff08;特别是由托管平台/服务通过 API 调用公开的大型语言模型&#xff0c;如 Claude 的 LLama 2等&#xff09;&#xff0c…

创邻科技张晨:期待解锁图技术在供应链中的关联力

近日&#xff0c;创邻科技创始人兼CEO张晨博士受浙江省首席信息官协会邀请&#xff0c;参加数字化转型与企业出海研讨会。 此次研讨会旨在深入探讨数字经济时代下&#xff0c;企业如何有效应对成本提升与环境变化所带来的挑战&#xff0c;通过数字化转型实现提效增益&#xff…

解决 Visual C++ 17.5 __cplusplus 始终为 199711L 的问题

目录 软件环境问题描述查阅资料解决问题参考文献 软件环境 Visual Studio 2022, Visual C, Version 17.5.4 问题描述 在应用 https://github.com/ToniLipponen/cpp-sqlite 的过程中&#xff0c;发现源代码文件 sqlite.hpp 中&#xff0c;有一处宏&#xff0c;和本项目的 C L…

R语言数据分析案例27-使用随机森林模型对家庭资产的回归预测分析

一、研究背景及其意义 家庭资产分析在现代经济学中的重要性不仅限于单个家庭的财务健康状况&#xff0c;它还与整个经济体的发展紧密相关。家庭资产的增长通常反映了国家经济的整体增长&#xff0c;而资产分布的不均则暴露了经济不平等的问题。因此&#xff0c;全球视角下的家…

好像也没那么失望!SD3玩起来,Stable Diffusion 3工作流商业及广告设计(附安装包)

今天基于SD3 base 工作流来尝试进行下广告设计&#xff0c;这要是一配上设计文案&#xff0c;视觉感就出来了。下面来看看一些效果展示~ SD3 Medium模型及ComfyUI工作流下载地址&#xff1a;文末领取&#xff01; 1.清凉夏日——西瓜音乐会 提示词&#xff1a; a guitar wi…

基于springboot实现火锅店管理系统项目【项目源码+论文说明】

基于springboot实现火锅店管理系统演示 摘要 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff0c;在计算机上安装火锅店管理系统软件来…

LabVIEW软件开发任务的工作量估算方法

在开发LabVIEW软件时&#xff0c;如何准确估算软件开发任务的工作量。通过需求分析、功能分解、复杂度评估和资源配置等步骤&#xff0c;结合常见的估算方法&#xff0c;如专家判断法、类比估算法和参数估算法&#xff0c;确保项目按时按质完成&#xff0c;提供项目管理和资源分…

机器学习笔记 - 用于3D点云数据分类的Point Net的训练

一、数据集 ShapeNet 是一项持续不断的努力,旨在建立一个注释丰富的大型 3D 形状数据集。我们为世界各地的研究人员提供这些数据,以支持计算机图形学、计算机视觉、机器人技术和其他相关学科的研究。ShapeNet 是普林斯顿大学、斯坦福大学和 TTIC 研究人员的合作成果。 Shape…