Go -- 测试 and 项目实战

没有后端基础,学起来真是费劲,所以打算速刷一下,代码跟着敲一遍,有个印象,大项目肯定也做不了了,先把该学的学了,有空就跟点单体项目,还有该看的书....

目录

🍌单元测试

🌼assert

🌼覆盖率

🦊tips

🦊依赖

🦊文件处理

🍌Mock测试

🍌基准测试

🍉项目实战

🦂ER图

🦂分层结构

🦂组件工具

💪Repository

💪Service

💪Controller

💪Router

💤运行


🍌单元测试

规则

(1)测试文件以 _test.go 结尾

源代码与测试代码👇

(2)测试函数命名规范:func TestXxx(*testing.T)     Test紧挨着第一个字母大写👇

(3)初始化逻辑放到TestMain()里

🌼assert

导入开源网站的assert包,来进行判等,不等等测试操作👇

单元测试的样例👇

judge.go

package judgefunc JudgePassLine(score int16) bool {if score >= 60 {return true}return false
}

judge_test.go

package judgeimport ("github.com/stretchr/testify/assert""testing"
)func TestJudgePassLineTrue(t *testing.T) {isPass := JudgePassLine(70)assert.Equal(t, true, isPass)
}

Run一下👇

👇

👆解释

testing.T 是 Go 语言测试框架中的一个结构体类型,它提供了一组方法和属性用于管理和报告测试的结果。当你编写和运行测试函数时,测试框架会自动创建一个 testing.T 类型的实例,并将其传递给测试函数

🌼覆盖率

当我们写了单元测试后👇

覆盖率越高,代码的质量越有保证,那如何查看覆盖率呢👇

PS:发现个Goland小技巧,ctrl + z可以返回上一步代码,误删了也不要紧

要查看覆盖率,首先要切到对应目录下,视频中是👇

但我只会先cd到对应目录,结果一样👇

覆盖率意味着👇

judge.go,函数中一共3行有效代码,但是judge_test.go传入的70,只会运行前2行,所以覆盖率是 2 / 3

当我们需要100%覆盖率,只需要增加一个测试分支👇

func TestJudgePassLineFail(t *testing.T) {isPass := JudgePassLine(50)assert.Equal(t, true, isPass)
}

👆通过不断地,对各个分支的测试,保证了测试的完备性, 减少了BUG的产生

🦊tips

🦊依赖

DB:数据库database

Cache:Redis类似的组件

File:本地文件

👆三项属于项目中的强依赖

在单元测试中,一般有2个目标:(1)幂等  (2)稳定

(1)幂等:多次重复一个case的测试,结果一样

(2)稳定:单元测试之间是相互隔离的,即任何时间 / 函数下,都能够运行

而需要保证幂等 / 稳定,需要Mock机制,下面先讲文件处理

🦊文件处理

log

line11
line22
line33
line44
line55

ProcessFirstLine.go

package firstLineimport ("bufio""os""strings"
)// ReadFirstLine 从文件中读取第一行内容
func ReadFirstLine() string {open, err := os.Open("log") // 打开名为 "log" 的文件defer open.Close()          // 延迟关闭文件, 避免资源泄露if err != nil {return "" // 发生错误时返回空字符串}scanner := bufio.NewScanner(open) // 创建一个扫描器for scanner.Scan() {              // 循环遍历文件的每一行return scanner.Text() // 返回第一行内容}return "" // 文件为空时返回空字符串
}// ProcessFirstLine 处理第一行内容,将 "11" 替换为 "00"
func ProcessFirstLine() string {line := ReadFirstLine()                          // 调用 ReadFirstLine 函数获取第一行内容destLine := strings.ReplaceAll(line, "11", "00") // 将 "11" 替换为 "00"return destLine                                  // 返回替换后的结果
}

ProcessFirstLine_test.go

package firstLineimport ("github.com/stretchr/testify/assert""testing"
)// TestProcessFirstLine 是对 ProcessFirstLine 函数的单元测试
func TestProcessFirstLine(t *testing.T) {firstLine := ProcessFirstLine()      // 调用 ProcessFirstLine 函数assert.Equal(t, "line00", firstLine) // 使用断言验证结果是否符合预期
}

equal还是报错,但是点击这里还是可以PASS👇

解释

(1)bufio:高效读取数据的包

  • bufio.NewReader():创建一个新的Reader对象,用于读取数据。它接收一个io.Reader类型的参数,并使用默认大小的缓冲区
  • bufio.NewScanner():创建一个新的Scanner对象,用于对输入流进行扫描。Scanner可用于逐行读取文本等
  • Reader类型:它提供了各种方法来从输入流中读取数据,如ReadString()用于读取字符串,ReadBytes()用于读取字节切片等
  • Scanner类型:它提供了各种方法来解析输入流中的数据,如Scan()用于逐行读取文本,ScanBytes()用于逐个字节读取等
  • Scanner.Text():返回当前扫描的文本内容
  • Scanner.Scan():将扫描器移动到下一行,并返回是否还有更多行可供扫描

(2)os是一个提供与操作系统相关功能的包

  • os.Args:一个字符串切片,包含命令行参数
  • os.Exit(code int):终止当前程序的执行,并返回给定的错误码
  • os.Getwd():返回当前的工作目录的路径名
  • os.Chdir(dir string):将当前的工作目录更改为指定的目录
  • os.Mkdir(name string, perm FileMode) error:创建一个新目录
  • os.Open(name string) (*File, error):打开一个文件用于读取
  • os.Create(name string) (*File, error):创建一个文件用于写入
  • os.Remove(name string) error:删除指定的文件或目录
  • os.Rename(oldname, newname string) error:重命名(移动)文件或目录
  • os.Stdoutos.Stdinos.Stderr:标准输出、标准输入和标准错误输出的文件对象

但是,如果源文件被人篡改,那么测试文件,在特定场景下,也就无法运行,那如何Mock呢?👇

🍌Mock测试

开源Mock测试包 -- bouk/monkey: Monkey patching in Go (github.com)

打桩,即用函数A替换函数B👇

不得不吐槽一下,内部课贴的代码只贴部分,默认剩下部分你都会了,没学过也不能无中生有吧...真的觉得,每一点代码都力求自己得到对应输出,有点浪费时间了,先速通吧...

🍌基准测试

看了一遍...

🍉项目实战

描述

用例

👆用户浏览页面,主要展示2方面内容,一是Topic话题,一是PostList回帖的列表

🦂ER图

Entity Relationship Diagram

ER图由以下三个主要组成部分构成:实体,属性,关系。

🦂分层结构

数据层:数据Model,外部数据的增删改查

逻辑层:业务Entity

视图层:视图view,处理和外部的交互逻辑

🦂组件工具

(1)Gin高性能 go web 框架

gin-gonic/gin: Gin is a HTTP web framework written in Go (Golang). It features a Martini-like API with much better performance -- up to 40 times faster. If you need smashing performance, get yourself some Gin. (github.com)

(2)Go mod

go mod init

go get gopkg.in/gin-gonic/gin.v1@v1.3.0 --> 执行后,go.mod就有gin的依赖

💪Repository

index

两个索引👇

初始化话题数据索引👇

package main //入口import ("bufio"         //缓冲读取"encoding/json" //处理JSON数据"os"            //操作系统交互
)// 初始化主题索引映射
func initTopicIndexMap(filePath string) error {open, err := os.Open(filePath + "topic") //打开文件filePathif err != nil {                          //打开错误return err}scanner := bufio.NewScanner(open) //逐行扫描的工具topicTmpMap := make(map[int64]*Topic) //值是指向Topic结构体的指针for scanner.Scan() {text := scanner.Text() //保存每行内容var topic Topicif err := json.Unmarshal([]byte(text), &topic); err != nil {return err //json.Unmarshal将text解析为Topic结构体}topicTmpMap[topic.Id] = &topic //保存解析的结构体保}topicIndexMap = topicTmpMap //映射赋值给全局变量return nil
}

查询 

👇

package mainimport ("bufio"         //缓冲读取"encoding/json" //处理JSON数据"os"            //操作系统交互"sync"
)// 声明一个全局变量 topicIndexMap,用于存储主题索引映射
var topicIndexMap map[int64]*Topic// Topic 定义 Topic 结构体,包含了主题的属性
type Topic struct {Id         int64  `json:"id"`Title      string `json:"title"`Content    string `json:"content"`CreateTime int64  `json:"create_time"`
}// TopicDao 定义 TopicDao 结构体
type TopicDao struct {
}var (topicDao  *TopicDaotopicOnce sync.Once
)// NewTopicDaoInstance 函数返回一个 TopicDao 实例
func NewTopicDaoInstance() *TopicDao {topicOnce.Do(func() {topicDao = &TopicDao{}})return topicDao
}// QueryTopicById 方法根据 id 查询并返回对应的 Topic 实例
func (*TopicDao) QueryTopicById(id int64) *Topic {return topicIndexMap[id]
}// 初始化主题索引映射
func initTopicIndexMap(filePath string) error {open, err := os.Open(filePath + "topic") //打开文件filePathif err != nil {                          //打开错误return err}scanner := bufio.NewScanner(open)     //逐行扫描的工具topicTmpMap := make(map[int64]*Topic) //值是指向Topic结构体的指针for scanner.Scan() {text := scanner.Text() //保存每行内容var topic Topicif err := json.Unmarshal([]byte(text), &topic); err != nil {return err //json.Unmarshal将text解析为Topic结构体}topicTmpMap[topic.Id] = &topic //保存解析的结构体保}topicIndexMap = topicTmpMap //映射赋值给全局变量return nil
}

💪Service

定义两个实体👇

流程👇

流程代码👇

//Do 方法执行查询页面信息的流程,返回 PageInfo 实例和错误信息
func (f *QueryPageInfoFlow) Do() (*PageInfo, error) {if err := f.checkParam(); err != nil { //参数校验return nil, err //校验失败}if err := f.prepareInfo(); err != nil { //准备数据return nil, err}if err := f.packPageInfo(); err != nil { //组装实体return nil, err}return f.pageInfo, nil //返回 PageInfo 实例和 nil,操作成功
}

💪Controller

....看了一遍....

💪Router

💤运行

end....

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从Spring的角度看Memcached和Redis及操作

目录 Memcached和Redis的区别 适用场景 Memcached配置使用 Redis配置使用 在SpringBoot的框架里,有直连Redis的SDK却没有Memcached的,可见相比地位。不过各有各的适应场景,Redis这个单线程模型确实非常强。 Memcached和Redis的区别 共同…

redis的数据类型及操作

三、redis的数据类型 String字符串 set、get mset setex setnx 会检测键值对存不存在,如果存在不发生变化,如果存在则增加键值对 只增加 而set会覆盖原来的值 增加、修改 setrange 有下标则替换,没有则添加 getrange 获取全…

剑指 Offer 54. ! 二叉搜索树的第k大节点 (考察二叉树的中序遍历)

剑指 Offer 54. 二叉搜索树的第k大节点 给定一棵二叉搜索树,请找出其中第 k 大的节点的值。 我的思路是:用一个全局arrayList不断收集“逆向”中序遍历该搜索二叉树所需要的答案 class Solution {int res, k;public int kthLargest(TreeNode root, int …

为Stable Diffusion web UI开发自己的插件实战

最近,Stable Diffusion AI绘画受到了广泛的关注和热捧。它的Web UI提供了了一系列强大的功能,其中特别值得一提的是对插件的支持,尤其是Controlnet插件的加持,让它的受欢迎程度不断攀升。那么,如果你有出色的创意&…

GLM模型介绍

paper: 《GLM: General Language Model Pretraining with Autoregressive Blank Infilling》 摘要: 我们提出了一个基于自回归空白填充的通用语言模型(GLM)来解决这一挑战。GLM通过添加2D位置编码和允许任意顺序预测跨度来改进空白填充预训…

机器学习笔记之优化算法(四)线搜索方法(步长角度;非精确搜索)

机器学习笔记之优化算法——线搜索方法[步长角度,非精确搜索] 引言回顾:精确搜索步长及其弊端非精确搜索近似求解最优步长的条件反例论述 引言 上一节介绍了从精确搜索的步长角度观察了线搜索方法,本节将从非精确搜索的步长角度重新观察线搜…

蓝网科技股份有限公司存在SQL注入

书把他从沉重的生活中拉出来,使他的精神不致被劳动压的麻木不仁。通过不断地读书,他认识到,只有一个人对世界了解得更广大,对人生看得更深刻,那么,他才可能对自己所处的艰难和困苦有更高意义的理解&#xf…

AI 绘画Stable Diffusion 研究(五)sd文生图功能详解(下)

大家好,我是风雨无阻。 上一篇文章详细介绍了sd文生图的功能及使用注意事项,感兴趣的朋友可以前往查看:AI 绘画Stable Diffusion 研究(四)sd文生图功能详解(上) 。 那今天这篇文章,我…

论文阅读 - Social bot detection in the age of ChatGPT: Challenges and opportunities

论文链接:https://www.researchgate.net/publication/371661341_Social_bot_detection_in_the_age_of_ChatGPT_Challenges_and_opportunities 目录 摘要: 引言 1.1. Background on social bots and their role in society 1.2. The rise of AI-gene…

计算机是如何工作的(笔记)

目录 寄存器 操作系统 进程(process) CPU pcb中关于进程调度相关的属性 寄存器 用来存储数据的单位,是CPU的一部分 寄存器,存储空间更小,访问速度更快,成本更高,掉电后数据会丢失 寄存器…

java 定时任务不按照规定时间执行

这里写目录标题 使用异步启动可能出现的问题排查代码中添加的定时任务步骤是否正确排查是否任务阻塞,如果定时任务出现异常阻塞后,将不会在次执行java中多个Scheduled定时器不执行为了让Scheduled效率更高,我们可以通过两种方法将定时任务变成…

【LeetCode 75】第十五题(1456)定长子串中元音的最大数目

目录 题目: 示例: 分析: 代码运行结果: 题目: 示例: 分析: 就难度而言,我觉得算不上中等,因为和上一题基本一致,只不过上一题是求最大平均数&#xff0c…

大麦订单生成器 大麦订单购买截图生成

后台一键生成链接,独立后台管理 教程:修改数据库config/Conn.php 不会可以看源码里有教程 下载程序:https://pan.baidu.com/s/16lN3gvRIZm7pqhvVMYYecQ?pwd6zw3

过滤器,监听器与拦截器的区别

过滤器,监听器与拦截器的区别 ​ 过滤器和监听器不是Spring MVC中的组件,而是Servlet的组件,由Servlet容器来管理。拦截器是Spring MVC中的组件,由Spring容器来管理 ​ Servlet过滤器与Spring MVC 拦截器在Web应用中所处的层次如…

node.js系列-多种方案教你在node程序中同时使用CommonJS 和 ES Module 混合开发最佳实践

前情提要 我们平时使用的npm 第三方包一般基于这两种规范开发的,很容易遇到一个项目里既有 CommonJS 又有 ES Module 的情况,那么我们应该如何解决这种CommonJS 和 ES Module 混合开发的问题呢? CommonJS是什么? 2009年&#x…

MySQL主从复制原理以及实操

一、MySQL主从复制原理: 1、MySQL将数据变化记录到二进制日志中; 2、Slave将MySQL的二进制日志拷贝到Slave的中继日志中; 3、Slave将中继日志中的事件在做一次,将数据变化,反应到自身(Slave)的数…

【Linux】常用的基本指令

👦个人主页:Weraphael ✍🏻作者简介:目前正在学习c和算法 ✈️专栏:Linux 🐋 希望大家多多支持,咱一起进步!😁 如果文章有啥瑕疵,希望大佬指点一二 如果文章对…

C# Microsoft消息队列服务器的使用 MSMQ

先安装消息队列服务器 private static readonly string path ".\\Private$\\myQueue";private void Create(){if (!MessageQueue.Exists(path)){MessageQueue.Create(path);}}private void Send(){Stopwatch stopwatch new Stopwatch();stopwatch.Start();Message…

K8s的详细介绍

1.编写yaml文件的方式 2.yaml里面的内容介绍 Pod实现机制:(1)共享网络(2)共享存储 共享网络:通过Pause容器,把其他业务容器加入到Pause容器里面,让所有业务容器在同一个名称空间中,…

【Spring Cloud 三】Eureka服务注册与服务发现

系列文章目录 【Spring Cloud一】微服务基本知识 Eureka服务注册与服务发现 系列文章目录前言一、什么是Eureka?二、为什么要有服务注册发现中心?三、Eureka的特性四、搭建Eureka单机版4.1Eureka服务端项目代码pom文件配置文件启动类启动项目查看效果 E…