Go -- 测试 and 项目实战

news2025/1/12 12:04:19

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

目录

🍌单元测试

🌼assert

🌼覆盖率

🦊tips

🦊依赖

🦊文件处理

🍌Mock测试

🍌基准测试

🍉项目实战

🦂ER图

🦂分层结构

🦂组件工具

💪Repository

💪Service

💪Controller

💪Router

💤运行


🍌单元测试

规则

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

源代码与测试代码👇

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

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

🌼assert

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

单元测试的样例👇

judge.go

package judge

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

judge_test.go

package judge

import (
	"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 firstLine

import (
	"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 firstLine

import (
	"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") //打开文件filePath
	if err != nil {                          //打开错误
		return err
	}
	scanner := bufio.NewScanner(open) //逐行扫描的工具
	topicTmpMap := make(map[int64]*Topic) //值是指向Topic结构体的指针
	for scanner.Scan() {
		text := scanner.Text() //保存每行内容
		var topic Topic
		if err := json.Unmarshal([]byte(text), &topic); err != nil {
			return err //json.Unmarshal将text解析为Topic结构体
		}
		topicTmpMap[topic.Id] = &topic //保存解析的结构体保
	}
	topicIndexMap = topicTmpMap //映射赋值给全局变量
	return nil
}

查询 

👇

package main

import (
	"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  *TopicDao
	topicOnce 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") //打开文件filePath
	if err != nil {                          //打开错误
		return err
	}
	scanner := bufio.NewScanner(open)     //逐行扫描的工具
	topicTmpMap := make(map[int64]*Topic) //值是指向Topic结构体的指针
	for scanner.Scan() {
		text := scanner.Text() //保存每行内容
		var topic Topic
		if 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.coloradmin.cn/o/830705.html

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

相关文章

【LeetCode】【数据结构】单链表OJ常见题型(二)

👀樊梓慕:个人主页 🎥个人专栏:《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》 🌝每一个不曾起舞的日子,都是对生命的辜负 目录 前言: 【LeetCode】面试题02.04. 分割链表 【Lee…

Unity实现物体上下漂浮旋转效果

效果展示 代码(直接挂载在物体上就可以了) using System.Collections; using System.Collections.Generic; using UnityEngine;public class FloatingObject : MonoBehaviour {public float amplitude 0.5f; // 上下移动的幅度public float frequency …

C# OpenCvSharp 去水印 图像修复

效果 项目 VS2010.net4.0OpenCvSharp3 代码 using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.IO; using System…

简单了解一下vue-router是什么

要学习vue-router就要先知道这里的路由是什么&#xff1f;为什么我们不能像原来一样直接用<a></a>标签编写链接哪&#xff1f;vue-router如何使用&#xff1f;常见路由操作有哪些&#xff1f;等等这些问题&#xff0c;就是本篇要探讨的主要问题。 vue-router是什么…

【二叉树进阶】二叉树的前中后序遍历(非递归迭代实现)

文章目录 1. 二叉树的前序遍历1.1 思路分析1.2 AC代码 2. 二叉树的中序遍历2.1 思路分析2.2 AC代码 3. 二叉树的后序遍历3.1 思路13.2 思路1AC3.3 思路23.4 思路2AC 1. 二叉树的前序遍历 题目链接: link 不用递归&#xff0c;用迭代算法如何实现对二叉树的前序遍历&#xff1f…

linux作业

1.简述静态网页和动态网页的区别 (1).程序是否在服务器端运行&#xff0c;是重要标志。 (2).编程技术不同。静态网页和动态网页主要根据网页制作的语言来区分。 (3).被搜索引擎收录情况不同。 (4).用户访问速度不同。 (5).制作和后期维护工作量不同。 2. 简述 Webl.0 和 …

嵌入式pc技术的特点有哪些?

嵌入式PC技术是将计算机硬件和软件嵌入到各种设备中的一种技术&#xff0c;它具有低功耗、高效率、小型化、易于集成等优点&#xff0c;广泛应用于工业自动化、医疗设备、电力、通信、家用电器、物联网等领域&#xff0c;成为新时代工业生产和社会生活必不可少的技术之一。 嵌入…

Python基本数据类型之散列类型详解

前言&#xff1a; python的基本数据类型可以分为三类&#xff1a;数值类型、序列类型、散列类型&#xff0c;本文主要介绍散列类型。 一、散列类型 散列类型&#xff1a;内部元素无序&#xff0c;不能通过下标取值 1&#xff09;字典&#xff08;dict&#xff09;&#xff…

SAP 特殊采购类型52简介

特殊采购类型52简介-52 直接生产/收集订单可以在物料主数据中进行设置或者在BOM中进行设置, 所谓“直接生产”,是相对于一般的“间接生产”模式而言的。在我们通常采用的计划模式下面,所有在BOM结构里面的半成品,都是在库存中作为一个整体, 可以用在任意的一个成品物料,或…

STM32 NOR_FLASH 学习

NOR FLASH FLASH是常用的&#xff0c;用于存储数据的半导体器件&#xff0c;它具有容量大&#xff0c;可重复擦写、按“扇区/块”擦除、掉电后数据可继续保存的特性。 NOR FLASH的单位是MB&#xff0c;EEPROM的单位是KB。 NM25Q128&#xff0c;是NOR FLASH的一种&#xff0c…

正则匹配img标签里面src

正则&#xff1a; (?<src\s*\s*\")\S(?\"{1})匹配效果&#xff1a;

SAP F4下拉值报错:【内部错误:表格格式】

报错截图如下&#xff1a; 解决办法&#xff1a; 事务码&#xff1a;SU3 在【参数】页签维护如下值&#xff1a; SET/GET参数标识参数值简短描述F4METHODNoActiveXActiveX/NoActiveX 维护好以上信息之后&#xff0c;就可以正常显示下拉值了

OpenHarmony ArkUI 如何调用相机

​ ArkUI调用相机和调用相册其实是一个思路&#xff0c;只用修改一个地方。 我们继续来说相机调用&#xff0c;ArkUI没办法自己获取相机&#xff0c;所以得依靠一下ohos.multimedia.camera 相机开发指导 介绍 本指导主要展示了调用相机的调用过程&#xff0c;以及调用相机的…

【Elasticsearch】学好Elasticsearch系列-Query DSL

本文已收录至Github&#xff0c;推荐阅读 &#x1f449; Java随想录 先看后赞&#xff0c;养成习惯。 点赞收藏&#xff0c;人生辉煌。 文章目录 查询上下文相关度评分&#xff1a;_score源数据&#xff1a;_source数据源过滤器全文检索match&#xff1a;匹配包含某个term的子句…

整个个人博客?想找纯html代码模板?来个手机版带菜单的首页模板“参考参考”

以前做毕业设计的时候老想找一些不掺杂后端代码的前端模板。 可是下载下来&#xff0c;不是php就是python后台的。看又看不懂&#xff0c;想换语言就必须先把里面的后台代码拿掉。 就很像买了个精装的二手房&#xff0c;白白多花了砸墙钱。 就比如&#xff0c;想做个带菜单的…

【go-zero】docker镜像直接部署API与RPC服务 如何实现注册发现?docker network 实现 go-zero 注册发现

一、场景&问题 使用docker直接部署go-zero微服务会发现API无法找到RPC服务 1、API无法发现RPC服务 用docker直接部署 我们会发现API无法注册发现RPC服务 原因是我们缺少了docker的network网桥 2、系统内查看 RPC服务运行正常API服务启动,通过docker logs 查看日志还是未…

数据库相关知识点

体系结构图&#xff1a; 体系介绍&#xff1a; Client Connectors 接入方。支持很多协议(JDBC、ODBC、.NET、PHP、Python、PERL、C 等) Management Serveices & Utilities 系统管理和控制工具&#xff0c;mysqldump、 mysql复制集群、分区管理等 Connection Pool 连接池…

【css】css位置布局position

position 属性规定应用于元素的定位方法的类型。元素其实是通过使用top、bottom、left 和 right 属性来定位的。但是&#xff0c;需要首先设置了 position 属性&#xff0c;否则这些属性将不起作用。根据不同的 position 值&#xff0c;它们的设置特点不同。 其有五个不同的位…

【禅道】禅道数据迁移,源码安装的禅道18.2迁移至docker安装的禅道18.2

docker安装禅道开源版18.2 sudo docker run --name chandao \ -p 9080:80 \ -p 3306:3306 \ --networkzentaonet \ -v /opt/docker/zentao/zentaopms:/www/zentaopms \ -v /opt/docker/zentao/mysql:/var/lib/mysql \ -e MYSQL_ROOT_PASSWORD数据库密码\ -d easysoft/zentao:1…

Reset复位电路的PCB布局布线要求

Reset复位电路的PCB布局布线要求 —来源&#xff1a;瑞芯微RK3588 PCB设计白皮书 Reset复位电路是一种用来使电路恢复到起始状态的电路设计&#xff0c;一般简单的复位电路由电容串阻电阻构成&#xff0c;再复杂点就有三级管等配合进行&#xff0c;RK3588内置复位电路&#xf…