网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
包含单元测试和性能测试
通过go help test可以看到go test的使用说明:
格式形如:
go test [-c] [-i] [build flags] [packages] [flags for test binary]
参数解读:
-c : 编译go test成为可执行的二进制文件,但是不运行测试。
-i : 安装测试包依赖的package,但是不运行测试。
关于build flags,调用go help build,这些是编译运行过程中需要使用到的参数,一般设置为空
关于packages,调用go help packages,这些是关于包的管理,一般设置为空
关于flags for test binary,调用go help testflag,这些是go test过程中经常使用到的参数
-test.v : 是否输出全部的单元测试用例(不管成功或者失败),默认没有加上,所以只输出失败的单元测试用例。
-test.run pattern: 只跑哪些单元测试用例
-test.bench patten: 只跑那些性能测试用例
-test.benchmem : 是否在性能测试的时候输出内存情况
-test.benchtime t : 性能测试运行的时间,默认是1s
-test.cpuprofile cpu.out : 是否输出cpu性能分析文件
-test.memprofile mem.out : 是否输出内存性能分析文件
-test.blockprofile block.out : 是否输出内部goroutine阻塞的性能分析文件
-test.memprofilerate n : 内存性能分析的时候有一个分配了多少的时候才打点记录的问题。这个参数就是设置打点的内存分配间隔,也就是profile中一个sample代表的内存大小。默认是设置为512 * 1024的。如果你将它设置为1,则每分配一个内存块就会在profile中有个打点,那么生成的profile的sample就会非常多。如果你设置为0,那就是不做打点了。
你可以通过设置memprofilerate=1和GOGC=off来关闭内存回收,并且对每个内存块的分配进行观察。
-test.blockprofilerate n: 基本同上,控制的是goroutine阻塞时候打点的纳秒数。默认不设置就相当于-test.blockprofilerate=1,每一纳秒都打点记录一下
-test.parallel n : 性能测试的程序并行cpu数,默认等于GOMAXPROCS。
-test.timeout t : 如果测试用例运行时间超过t,则抛出panic
-test.cpu 1,2,4 : 程序运行在哪些CPU上面,使用二进制的1所在位代表,和nginx的nginx_worker_cpu_affinity是一个道理
-test.short : 将那些运行时间较长的测试用例运行时间缩短
目录结构
test
|
—— calc.go
|
—— calc_test.go
//calc.go
package main
func add(a, b int) int {
return a + b
}
func sub(a, b int) int {
return a - b
}
//calc_test.go
package main
import (
“testing”
)
func TestAdd(t *testing.T) {
r := add(2, 4)
if r != 6 {
t.Fatalf(“add(2, 4) error, expect:%d, actual:%d”, 6, r)
}
t.Logf(“test add succ”)
}
结果
cd test/
ls
calc.go calc_test.go
//-v参数显示通过函数的信息
yugoMBP:test yuchao$ go test -v
=== RUN TestAdd
— PASS: TestAdd (0.00s)
calc_test.go:11: test add succ…is ok
PASS
ok gostudy/gobook/test 0.006s
1.文件名必须是_test.go结尾的,这样在执行go test的时候才会执行到相应的代码
2.你必须import testing这个包
3.所有的测试用例函数必须是Test开头
4.测试用例会按照源代码中写的顺序依次执行
5.测试函数TestXxx()的参数是testing.T,我们可以使用该类型来记录错误或者是测试状态
6.测试格式:func TestXxx (t *testing.T),Xxx部分可以为任意的字母数字的组合,
但是首字母不能是小写字母[a-z],例如Testintdiv是错误的函数名。
7.函数中通过调用testing.T的Error, Errorf, FailNow, Fatal, FatalIf方法,说明测试不通过,调用Log方法用来记录测试的信息。
一个正整数的阶乘(factorial)是所有小于及等于该数的正整数的积,并且0的阶乘为1。自然数n的阶乘写作n!。1808年,基斯顿·卡曼引进这个表示法。
package main
import “fmt”
func factorial(i int) int {
if i <= 1 {
return 1
}
return i * factorial(i-1)
}
func main() {
var i int = 7
fmt.Printf(“Factorial of %d is %d\n”, i, factorial(i))
}
//输出结果:
Factorial of 7 is 5040
这个数列从第3项开始,每一项都等于前两项之和。
package main
import “fmt”
func fibonaci(i int) int {
if i == 0 {
return 0
}
if i == 1 {
return 1
}
return fibonaci(i-1) + fibonaci(i-2)
}
func main() {
var i int
for i = 0; i < 10; i++ {
fmt.Printf(“%d\n”, fibonaci(i))
}
}
//输出结果:
0
1
1
2
3
5
8
13
21
34
=================================================================================
程序开发中经常要创建资源(数据库初始化连接,文件句柄,锁等),在程序执行完毕都必须得释放资源,Go提供了defer(延时机制)更方便、更及时的释放资源。
-
内置关键字defer 用于延迟调用
-
defer在
return
前执行,常用于资源释放 -
多个defer按
先进后出
的机制执行 -
defer语句的变量,在defer声明时定义
-
显式 return 返回前,会
先修改命名返回参数
defer用途:
-
关闭
文件
句柄 -
锁
资源释放 -
数据库
连接释放
defer 是先进后出
后面的语句会依赖
前面的资源,因此如果先前面的资源先释放了
,后面的语句就没法执行
了。
package main
import “fmt”
func main() {
var whatever [5]struct{}
for i := range whatever {
defer fmt.Println(i)
}
}
//结果
4
3
2
1
0
package main
import (
“fmt”
)
func testDefer1() {
//defer机制 先入后出,如同羽毛球筒
defer fmt.Println(“hello v1”) //顺序5
defer fmt.Println(“hello v2”) //顺序4
defer fmt.Println(“hello v3”) //顺序3
fmt.Println(“aaaaa”) // 顺序1
fmt.Println(“bbbb”) //顺序2
}
func testDefer2() {
for i := 0; i < 5; i++ {
//每次循环,defer将后入的语句压到defer栈
//依旧先入后出
defer fmt.Printf(“i=%d\n”, i)
}
fmt.Printf(“running\n”) //顺序1
fmt.Printf(“return\n”) //顺序2
}
func testDefer3() {
var i int = 0
//defer是声明时定义好的,之后再修改无效,因此i=0
defer fmt.Printf(“defer i=%d\n”, i)
i = 1000
fmt.Printf(“i=%d\n”, i)
}
func main() {
//testDefer1()
//testDefer2()
testDefer3()
}
如果defer遇到了闭包
package main
import “fmt”
func main() {
var whatever [3]struct{}
for i := range whatever {
defer func() { fmt.Println(i) }()
}
}
//相当于
i结果最后为2,因为闭包原因,所以i最终为2
//结果,
2
2
2
package main
import “fmt”
type Test struct {
name string
}
func (t *Test) Close() {
fmt.Println(t.name, " closed")
}
func main() {
ts := []Test{{“a”}, {“b”}, {“c”}}
for _, t := range ts {
defer t.Close()
}
}
//预计的输出c b a,而是输出c c c
//结果
c closed
c closed
c closed
将defer t.Close()改为defer Close(t)
输出结果:
c closed
b closed
a closed
省去函数,可以用t.Close()
package main
import “fmt”
type Test struct {
name string
}
func (t *Test) Close() {
fmt.Println(t.name, " closed")
}
func main() {
ts := []Test{{“a”}, {“b”}, {“c”}}
for _, t := range ts {
t2 := t
defer t2.Close()
}
}
c closed
c closed
c closed
结论:
defer后面的语句在执行的时候,函数调用的参数会被保存起来
,但是不执行
。也就是复制了一份。但是并没有说struct这里的this指针如何处理,通过这个例子可以看出go语言并没有
把这个明确写出来的this指针当作参数
来看待。
多个 defer 注册,按 FILO
次序执行 ( 先进后出 )。
异常也会输出
package main
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
= []Test{{“a”}, {“b”}, {“c”}}
for _, t := range ts {
t2 := t
defer t2.Close()
}
}
c closed
c closed
c closed
结论:
defer后面的语句在执行的时候,函数调用的参数会被保存起来
,但是不执行
。也就是复制了一份。但是并没有说struct这里的this指针如何处理,通过这个例子可以看出go语言并没有
把这个明确写出来的this指针当作参数
来看待。
多个 defer 注册,按 FILO
次序执行 ( 先进后出 )。
异常也会输出
package main
[外链图片转存中…(img-M6EQWR9j-1715142810780)]
[外链图片转存中…(img-eVgUlNp7-1715142810780)]
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!