golang单元测试及mock总结
文章目录
- 一、前言
- 1、单测的定位
- 2、vscode中生成单测
- 二、构造测试case的注意事项
- 1、项目初始化
- 2、构造空interface{}
- 3、构造结构体的time.Time类型
- 4、构造json格式的test case
- 三、运行单测文件
- 1、整体运行单测文件
- 2、运行单个单测文件报错
- (1)command-line-arguments是什么
- (2)undefined发生原因
- (3)缺少初始化导致的发生panic
- 3、查看单测覆盖率
- 4、单测覆盖文件解读
- 5、生成可被浏览器打开的单测文件
- 6、单测覆盖率的问题
- 四、关于单测粒度的问题
- 1、chatgpt的回答
- 2、个人理解
- 五、mock数据
- 1、mock组件选择
- 2、mock实操
- (1)mock函数调用
- (2)mock方法调用
- (3)mock其他包的函数
- (4)mock循环中的函数
- (5)mock http调用
- 3、对于mock的看法
一、前言
1、单测的定位
单测在软件工程中的地位毋庸置疑,它要求工程师必须去主动思考代码的边界,异常处理等等。另一方面,它又是代码最好的说明书,你的函数具体做了什么,输入和输出一目了然。
计算机科学家Edsger Dijkstra曾说过:“测试能证明缺陷存在,而无法证明没有缺陷。”再多的测试也不能证明一个程序没有BUG。在最好的情况下,测试可以增强我们的信心:代码在很多重要场景下是可以正常工作的。
参考:go语言圣经之测试函数
2、vscode中生成单测
参考:在 VS Code 快速生成单元测试
vscode生成单元测试如下,我们需要编写测试用例数组,明确指出来want结果以及wantErr,通过遍历的方式去执行测试用例数组。
func TestGenerateStsTokenService(t *testing.T) {type args struct {ctx context.ContextgenerateStsData *dto.GenerateStsReqParams}tests := []struct {name stringargs argswantResp *common.RESTRespwantErr bool}{{name: "测试正常生成sts",args: args{ctx: context.TODO(),generateStsData: &dto.GenerateStsReqParams{SessionName: "webApp",AuthParams: &dto.AuthParamsData{},},},wantResp: &common.RESTResp{Code: 0,Data: &dto.OssStsRespData{},},wantErr: false,},{name: "测试异常生成sts",args: args{ctx: context.TODO(),generateStsData: &dto.GenerateStsReqParams{SessionName: "liteApp",AuthParams: &dto.AuthParamsData{},},},wantResp: &common.RESTResp{Code: 20003,Data: interface{}(nil),},wantErr: true,},}for _, tt := range tests {t.Run(tt.name, func(t *testing.T) {gotResp, err := GenerateStsTokenService(tt.args.ctx, tt.args.generateStsData)if (err != nil) != tt.wantErr {t.Errorf("GenerateStsTokenService() error = %v, wantErr %v", err, tt.wantErr)return}if !reflect.DeepEqual(gotResp, tt.wantResp) {t.Errorf("GenerateStsTokenService() = %v, want %v", gotResp, tt.wantResp)}})}
}
二、构造测试case的注意事项
1、项目初始化
// TestMain会在执行其他测试用例的时候,自动执行
func TestMain(m *testing.M) {setup() //初始化函数retCode := m.Run() // 运行单元测试teardown() //后置校验,钩子函数,可不实现os.Exit(retCode) //清理结果
}
2、构造空interface{}
// 直接给Data赋值为nil的话,验证会失败,
// 单纯的nil和(*infra.QueryOneMappingCode)(nil)是不一样的
wantResp: &common.RESTResp{Code: 0,Message: "",Data: (*infra.QueryOneMappingCode)(nil),},// 数组类型的空
// []dto.OneMappingCode{}也会验证失败
wantRes: []dto.OneMappingCode(nil),
3、构造结构体的time.Time类型
Data: &infra.xxx{ID: 54,Code: "338798",TakerUid: "",State: 1,Type: 1,CreatedAt: time.Date(2023, time.June, 9, 16, 32, 59, 0, time.Local),},也可以直接打印接口的返回,看看CreatedAt返回的是什么,然后构造一下就可以。
t.Logf("gotResp:(%#v)", gotResp.Data)
4、构造json格式的test case
wantResp: &common.RESTResp{Code: 0,Message: "success",Data: `{"id": 54,"code": "338798","creator_uid": "12345","client_appId": "1234","taker_uid": "","state": 1,"type": 1,"created_at": "2023-06-09T16:32:59+08:00"}`,},
三、运行单测文件
1、整体运行单测文件
cd /xxx 单测目录go test成功输出:PASSok
2、运行单个单测文件报错
错误提示如下:
# command-line-arguments [command-line-arguments.test]
./base_test.go:26:18: undefined: Ping
明明Ping函数和单测文件都在同一个包下面,为什么会出现undefined呢?command-line-arguments是什么?
答:
(1)command-line-arguments是什么
go test [flags] [packages] [build flags] [packages]
命令行参数中指定的每个包或文件都将被视为一个要进行测试的包。而 "command-line-arguments"
这个标识符就是用来表示上述情况中命令行参数中指定的文件。这样可以使 go test 命令将指定的文件作为单独的包进行处理,并执行其中的测试函数。
(2)undefined发生原因
错误提示build失败,也就是说我们需要把单测文件依赖的文件也传入进去。比如我这里单测base_test.go文件,则需要把base.go也写到命令行参数中。
具体参考:【Golang】解决Go test执行单个测试文件提示未定义问题
go test ./base.go ./base_test.go
(3)缺少初始化导致的发生panic
一般来说我们在一个package下,定义一个TestMain()函数就可以了,进行代码的初始化。但是当我们需要运行单个测试文件的时候,有可能这个测试文件里面恰好没有TestMain()了咋整。
api_test.goTestMain()
base_test.go // 没有TestMain()函数// 解决方案
1、初始化代码放到setup()函数中
2、go命令行
go test ./base.go ./base_test.go ./api_test.go ./api.go
3、只想运行base_test.go怎么办base_test.go中加上自己的setuoBase()
3、查看单测覆盖率
go test -covercoverage: 80.4% of statements
4、单测覆盖文件解读
go test -coverprofile=coverage.out// 打开单测覆盖率文件
mode: set
base.go:10.118,14.23 3 1
base.go:14.23,17.3 2 1解释如下:10.118,14.23 3 1 表示第 10 行到第 14 行代码被测试覆盖到了,且覆盖率为 3/1 (即 300%)。这是因为第 10 行至少执行了一次,如果执行了三次,则覆盖率为 300%。14.23,17.3 2 1 表示第 14 行到第 17 行代码被测试覆盖到了,且覆盖率为 2/1 (即 200%)。
5、生成可被浏览器打开的单测文件
go test -coverprofile=coverage.out
go tool cover -html=coverage.out -o coverage.html
绿色代表被覆盖到的代码,红色代表没有被覆盖到的代码。
左上角是运行单测命令目录下,所有go文件的覆盖率。
可以考虑新增单测case来覆盖到这部分红色。

6、单测覆盖率的问题
覆盖率为 100% 表示测试用例覆盖了所有的可能执行路径,即程序的所有功能都被覆盖到了。而覆盖率高于 100% 则表示相同的代码路径被多次测试或某些代码行在被测试期间被执行了多次。
但是单测100%并不能保证没有bug,只能保证写出来的代码没问题,但逻辑或者业务上的漏洞是检测不到的。
博主在滴滴的组是建议单测覆盖率50%以上,其他朋友的公司要求核心接口必须有单测,整体单测覆盖率30%以上。有需要的可以参考下。
四、关于单测粒度的问题
写单测的时候,总会疑问到底要写的多细呢?特别是原来项目没有单测的时候,补单测的代码比业务逻辑代码还多。。。
本例中,目录结构如下:
domain:base.gocode.gocode_test.goutil.go
code.go会调用base.go和util.go的函数,运行code_test.go发现单测覆盖率
已经80%了,是不是意味着只需要写个code_test.go就可以了呢?
1、chatgpt的回答
实际上不是的,base.go和util.go后续还可能被其他的文件使用,我们写单测的时候,应该尽量覆盖所有的异常情况,也就是程序的边界问题。因此base.go和util.go也需要做对应的单测,这样才能得到高质量的代码。
2、个人理解
单个code_test.go文件导致的问题是下层函数不mock,可能会影响到实际的数据,导致单测只能运行一次,而不能一直PASS。其次是代码流程变长导致单测case越写越多,接近集成测试了,这不是我们单测的目标。
把code_test.go中关于base.go和util.go的函数都给mock掉,发现单测覆盖率只有37%,且测试路径比较短。还需要分别写base_test.go和util_test.go,写完util_test.go单测覆盖率立马82%。
拆分的粒度变细,更加关注每个函数的输入和输出。特别是当修改某个函数的时候,只需要使用对应的单测来进行验证,而不需要从入口处进行测试。毕竟单元测试不是集成测试。
参考:
Golang 单元测试:有哪些误区和实践?
Go的单元测试技巧
五、mock数据
在写单测的时候,程序难免会出现各种跨文件的函数调用,以及操作第三方中间件或者上下游交互的情况,这个时候mock就显得尤为重要。
想象下,没有mock的时候,我们运行单测可能就会写入一次数据库?或者对下游发起一次请求?这样的单测,怕是只能运行一次哟。mock的出现让我们关注代码的实现细节,不会担心会造成数据污染或者单测只能运行一遍就GG的情况。
1、mock组件选择
参考:如何做好单元测试?Golang Mock”三剑客“ gomock、monkey、sqlmock
GO进阶单元测试

博主这里更喜欢无侵入的mock,直接一把梭。可惜monkey已经不更新了,现在都是用gomonkey,国人大佬开发的
gomonkey 项目库
解析 Golang 测试(8)- gomonkey 实战
2、mock实操
(1)mock函数调用
函数中存在大量的封装调用,比如A->B,A->C这种,因此自由mock B和C函数对我们的单元测试来说还是很重要的。
patches := gomonkey.ApplyFunc(queryOneMappCode, func(ctx context.Context, code string) (*infra.QueryOneMappingCode, error) {// 参数大于6则返回空if len(code) > 6 {return nil, nil}return &infra.QueryOneMappingCode{ID: 54,Code: "338798",CreatedAt: time.Date(2023, time.June, 9, 16, 32, 59, 0, time.Local),}, nil})defer patches.Reset()
(2)mock方法调用
1、实例化接口
var mockProvider = provider.Test
// 接口如下
type TestDbProvider interface {SetDb(db *sqlx.DB)GetOne(dest interface{}, sql string, args interface{}) (resp *infra.QueryOneMappingCode, err error)
}2、mock对应的查询方法
// 注意,第一个参数不能是指针,不然mock会失效
// 例如 var oss_bucket_obj *oss.Bucket ,传入target为: *oss_bucket_obj
// 传地址会报错
patches := gomonkey.ApplyMethodFunc(mockProvider, "GetOne", func(dest interface{}, sql string, args interface{}) (resp *infra.QueryOneMappingCode, err error) {code := args.(string)if code == "123456" {return &infra.QueryOneMappingCode{ID: 1,Code: "123456",CreatedAt: time.Date(2023, time.June, 9, 16, 32, 59, 0, time.Local),}, nil} else if code == "456789" {return &infra.QueryOneMappingCode{ID: 1,Code: "456789",CreatedAt: time.Date(2023, time.June, 9, 16, 32, 59, 0, time.Local),}, nil} else {return nil, nil}})defer patches.Reset()
(3)mock其他包的函数
在xx_test文件中直接引用其他包即可。一般xx_test.go和xx.go在同一个包下,所以也不用担心出现循环引用的问题。
patches := gomonkey.ApplyFunc(util.GenerateRandomCode, func(numDigits int) string {return "123456"})defer patches.Reset()
(4)mock循环中的函数
比如在A函数中,循环3次调用了B函数,那么mock如下:
createA := &infra.CreateMappingCode{Code: "933903"}createB := &infra.CreateMappingCode{Code: "601690"}createC := &infra.CreateMappingCode{Code: "798493"}p := gomonkey.ApplyFuncSeq(structureMappingCodeRecord, []gomonkey.OutputCell{{Values: gomonkey.Params{createA}},{Values: gomonkey.Params{createB}},{Values: gomonkey.Params{createC}},})defer p.Reset() // 恢复原始函数
(5)mock http调用
// vscode自动生成的test代码
for _, tt := range tests {t.Run(tt.name, func(t *testing.T) {// mock httptestts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {if r.Method != http.MethodGet {w.WriteHeader(http.StatusNotFound)}// 构造返回参数w.WriteHeader(http.StatusOK)// 获取POST请求的参数,根据参数返回不同的响应bodyBytes, err := io.ReadAll(r.Body)if err != nil {// 处理错误w.WriteHeader(http.StatusBadRequest)}// 获取post参数params := new(dto.GenerateStsReqParams)json.Unmarshal(bodyBytes, params)// 根据传递的参数返回不同的响应res := new(common.RESTResp)if params.SessionName == "webApp" {res = &common.RESTResp{Code: 0,Message: "success",Data: &dto.OssStsRespData{Region: "hangzhou",Bucket: "test",},}} else {res = &common.RESTResp{Code: 1,Message: "failed",Data: &dto.OssStsRespData{},}}// 模拟接口的返回,http接口返回是字节数据,因此需要json.MarshaljsonStr, _ := json.Marshal(res)w.Write(jsonStr)}))defer ts.Close()// 替换原来的url为mock的urlGenerateOssStsUrl = ts.URL// 发起请求,请求中的http会被mock掉gotResp, err := GenerateStsTokenService(tt.args.ctx, tt.args.generateStsData)if (err != nil) != tt.wantErr {t.Errorf("GenerateStsTokenService() error = %v, wantErr %v", err, tt.wantErr)return}t.Logf("gotResp:(%#v) ,wantResp:(%#v)", gotResp, tt.wantResp)if !reflect.DeepEqual(gotResp, tt.wantResp) {t.Errorf("GenerateStsTokenService() = %v, want %v", gotResp, tt.wantResp)}})}
3、对于mock的看法
对于mock,有以下两种态度
一方的人主张不要滥用mock,能不mock就不mock。被测单元也不一定是具体的一个
函数,可能是多个函数本来就应该串起来,必要的时候再mock。一方则主张将被测函数所有调用的外面函数全部mock掉,只关注被测函数自己的
一行行代码,只要调用其他函数,全都mock掉,用假数据来测试。
本来处于懒惰和少写单测的角度,我是支持第一种方式的。
例如:
单测函数:A函数
内部逻辑:A->B : B函数全是业务逻辑A->C : C函数包括mysql或者redis操作A->D->E: D函数纯业务逻辑,构造请求参数。E函数对外发起http请求
第一种方式是只mock C和E函数,测试A函数的时候,会把B和D也测试到。主打一个省事快捷。
直到我遇到了更复杂的场景,B里面还有B1和B2函数,D里面有D1和D2函数,逻辑非常复杂的情况下,第一种方式就变成了集成测试。单测用例慢慢变成了测试用例。 比如只修改D2函数的情况下,要修改和通过单测A进行测试。。。。
第二种方式,就是在每一层都mock掉外部调用。单测A就只关注A的逻辑,mock掉B,C,D,E,只关注B,C,D,E输出是正确或者错误的情况。
针对B,C,D,E函数又有自己的单测函数,充分覆盖掉。这样当修改D2函数的时候,只需要修改和通过D2的单测即可。
对于外部依赖,比如第三方库mysql,redis,mq这种统一进行mock。 对于内部的函数调用,建议是粒度细一些,A_test.go就只对A.go里面的逻辑负责。至于调用B.go的部分,就交给B_test.go吧。
end
相关文章:
golang单元测试及mock总结
文章目录 一、前言1、单测的定位2、vscode中生成单测 二、构造测试case的注意事项1、项目初始化2、构造空interface{}3、构造结构体的time.Time类型4、构造json格式的test case 三、运行单测文件1、整体运行单测文件2、运行单个单测文件报错(1)command-l…...
mysql中的‘\G’ ‘\g’ ‘;’ navicat dbeaver
省流: 在navicat、dbeaver等客户端中使用时,“\G”、“\g”、“;”都可以不需要。 “\G”、“\g”、“;”都是用来做sql的结束符用。“\g”、“;”作用完全等价。“\G”是将字段横排显示转换成纵列显示。 横排显示: id |e…...
驱动day4work
头文件 #ifndef __CKR_H__ #define __CKR_H__typedef struct {unsigned int MODER; // 00unsigned int OTYPER; // 04unsigned int OSPEEDR; // 08unsigned int PUPDR; // 0Cunsigned int IDR; // 10unsigned int ODR; // 14 } gpio_t;// GPIO口 #define PHY_GPI…...
[SQL挖掘机] - 字符串函数 - length
介绍: length函数是mysql中用于获取字符串长度的函数。它接受一个字符串作为参数,并返回该字符串的字符数量(包括空格和特殊字符)。 用法: 以下是length函数的语法: length(string)其中,string是要计算长度的字符串…...
「深度学习之优化算法」(十七)灰狼算法
1. 灰狼算法简介 (以下描述,均不是学术用语,仅供大家快乐的阅读) 灰狼算法(Grey Wolf Algorithm)是受灰狼群体捕猎行为启发而提出的算法。算法提出于2013年,仍是一个较新的算法。目前为止(2020)与之相关的论文也比较多,但多为算法的应用,应该仍有研究和改进的余…...
mysql主从复制(主-从-从)
文章目录 一、前期环境准备二、主库配置1.设置server-id值并开启binlog参数2.建立同步账户并给上权限3.查看主库状态4.锁表设置只读5.备份数据库数据 三、从库配置1.设置server-id值并开启binlog参数2.还原从主库备份数据3.设定从主库同步4.启动从库同步开关 四.测试1.在主库上…...
如何制定数据采集解决方案?
数据采集仍是人工智能(AI)构建团队的主要瓶颈。原因各不相同:用例数据可能不足,深度学习等新机器学习(ML)技术需要更多数据,或者团队并未建立获取所需数据的适当流程。但无论如何,对…...
RabbitMQ消息可靠性问题及解决
说明:在RabbitMQ消息传递过程中,有以下问题: 消息没发到交换机 消息没发到队列 MQ宕机,消息在队列中丢失 消息者接收到消息后,未能正常消费(程序报错),此时消息已在队列中移除 …...
2023河南萌新联赛第(三)场:郑州大学(两个题目)
1.入门mex 重点 一些数字的mex是从0往上枚举,第一个没出现的数字。请你回答选最多k个数字,mex最大是多少 既然从0开始枚举,那么应该是最小,那么最大是什么? 经过自己的考虑,给出一个样例,0 1 1…...
学生管理系统-07打包与上线
一、项目架构 vue的项目必须要进行打包,并部署在nginx服务器上的 二、vue的打包 1、修改vue.cofing.js文件 在该文件中添加publicPath属性,值为./ const { defineConfig } require(vue/cli-service) module.exports defineConfig({transpileDepen…...
day31贪心算法 用最少数量的箭引爆气球 和无重叠区间
题目描述 题目分析: x轴向上射箭,12一支,重叠的需要一支,3-8一支,7-16一支 返回2; 就是让重叠的气球尽量在一起,局部最优;用一支弓箭,全局最优就是最少弓箭;…...
AMEYA360报道:手机直连卫星通信发展的三个阶段
卫星通信的发展从过去、现在与规划,可以分为三个阶段。手机卫星通信的第一个阶段中,较为典型的有铱星公司、海事卫星电话、天通卫星通信等,终端设备方面已经可以做到手持设备直接通过自带的天线与卫星进行通信。 包括铱星、天通卫星等&#x…...
redis中缓存雪崩,缓存穿透,缓存击穿的原因以及解决方案
一 redis的缓存雪崩 1.1 缓存雪崩 在redis中,新,旧数据交替时候,旧数据进行了删除,新数据没有更新过来,造成在高并发环境下,大量请求查询redis没有数据,直接查询mysql,造成mysql的…...
ChatGPT火热之下的冷思考
作为一款基于人工智能的自然语言处理(NLP)聊天机器人程序,ChatGPT通过大量来自互联网的文本进行训练,并使用深度学习和机器学习算法来理解用户的问题并提供准确的回答。并且,ChatGPT还内置了情感分析、关键字提取和实体识别等功能&am…...
查看docker容器启动参数
查看docker启动参数 1、查看docker容器的自启动策略2、查看docker容器的日志滚动清理策略 以下配置命令以redis容器为例 1、查看docker容器的自启动策略 docker inspect --format{{json .HostConfig.RestartPolicy}} redis输出的name是always 表示此容器是开机自启动的&#x…...
对Webpack的理解
Webpack是目前比较物流的前端构建工具,它基于入口,用不同的Loader来处理不同的文件 Webpack的核心概念 Entry:入口,Webpack执行构建的第一步将从Entry开始,可抽象成输入。告诉Webpack要使用哪个模块作为构建项目的起…...
使用wxPython和pillow开发拼图小游戏(四)
上一篇介绍了使用本地图片来初始化游戏的方法,通过前边三篇,该小游戏的主要内容差不多介绍完了,最后这一篇来介绍下游戏用时的计算、重置游戏和关闭窗口事件处理 游戏用时的计算 对于游戏用时的记录,看过前几篇的小伙伴可能也发现…...
XGBoost实例——皮马印第安人糖尿病预测和特征筛选
利用皮马印第安人糖尿病数据集来预测皮马印第安人的糖尿病,以下是数据集的信息: Pregnancies:怀孕次数Glucose:葡萄糖BloodPressure:血压 (mm Hg)SkinThickness:皮层厚度 (mm)Insulin:胰岛素 2…...
使用MQ发送对象错误
说明:使用RabbitMQ发送消息,消息是对象,出现下面这样的错误; 错误信息:Caused by: com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance of com.hmall.item.pojo.Item (no Cr…...
安装和卸载docker,详细教程
安装docker ############################################################################# 安装: 1、Docker要求CentOS系统的内核版本高于 3.10 ,通过 uname -r 命令查看你当前的内核版本是否支持安账docker 2、更新yum包:sudo yum -y up…...
【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型
摘要 拍照搜题系统采用“三层管道(多模态 OCR → 语义检索 → 答案渲染)、两级检索(倒排 BM25 向量 HNSW)并以大语言模型兜底”的整体框架: 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后,分别用…...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
模型参数、模型存储精度、参数与显存
模型参数量衡量单位 M:百万(Million) B:十亿(Billion) 1 B 1000 M 1B 1000M 1B1000M 参数存储精度 模型参数是固定的,但是一个参数所表示多少字节不一定,需要看这个参数以什么…...
基于uniapp+WebSocket实现聊天对话、消息监听、消息推送、聊天室等功能,多端兼容
基于 UniApp + WebSocket实现多端兼容的实时通讯系统,涵盖WebSocket连接建立、消息收发机制、多端兼容性配置、消息实时监听等功能,适配微信小程序、H5、Android、iOS等终端 目录 技术选型分析WebSocket协议优势UniApp跨平台特性WebSocket 基础实现连接管理消息收发连接…...
定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
让AI看见世界:MCP协议与服务器的工作原理
让AI看见世界:MCP协议与服务器的工作原理 MCP(Model Context Protocol)是一种创新的通信协议,旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天,MCP正成为连接AI与现实世界的重要桥梁。…...
【HTTP三个基础问题】
面试官您好!HTTP是超文本传输协议,是互联网上客户端和服务器之间传输超文本数据(比如文字、图片、音频、视频等)的核心协议,当前互联网应用最广泛的版本是HTTP1.1,它基于经典的C/S模型,也就是客…...
什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
论文笔记——相干体技术在裂缝预测中的应用研究
目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术:基于互相关的相干体技术(Correlation)第二代相干体技术:基于相似的相干体技术(Semblance)基于多道相似的相干体…...
