前言
在Golang1.21这一次更新中,主要更新内容为:
-
for range的一个语义变更 、
-
新加入max、min、clear方法、
-
contenxt增添api、
-
WASI的支持
本文主要带大家熟悉这些变更的内容~
1.for语义的变更
首先在go中for range其实有一个比较坑的地方
举一个简单的例子
go
printRes:=[]func(){}
for _,v:=range []int{1,2,3,4,5}{
printRes=append(printRes,func(){
fmt.Print(v)
})
}
for _,v:=range printRes{
v()
}
不知道for这个坑的同学可能会认为结果应该为12345
,但实际上它的结果是55555
其原因就是这里的循环遍历是使循环变量每次循环 。 循环遍历始终是同一个v这才导致了最后的结果都是一致的,要想达到使循环变量每次迭代而不是每次循环 其实也不难,只需要在每次循环时都新变量去重新赋值就好了v:=v
而1.21也是决定了重新定义循环的语义,让其是每次迭代。在1.21当中我们只需要在运行时额外加一个 GOEXPERIMENT=loopvar
构建程序即可
2.新增max,min,clear函数
在1.21版本go也是基于泛型新增了max
与min
函数,支持多种类型,多参数,在以前都需要自己去写这两个方法~
带大家看一下源码
go
// The max built-in function returns the largest value of a fixed number of
// arguments of [cmp.Ordered] types. There must be at least one argument.
// If T is a floating-point type and any of the arguments are NaNs,
// max will return NaN.
func max[T cmp.Ordered](x T, y ...T) T
// The min built-in function returns the smallest value of a fixed number of
// arguments of [cmp.Ordered] types. There must be at least one argument.
// If T is a floating-point type and any of the arguments are NaNs,
// min will return NaN.
func min[T cmp.Ordered](x T, y ...T) T
go
//支持的参数类型
type Ordered interface {
~int | ~int8 | ~int16 | ~int32 | ~int64 |
~uint | ~uint8 | ~uint16 | ~uint32 | ~uint64 | ~uintptr |
~float32 | ~float64 |
~string
}
举例
go
fmt.Println(max(1,uint(7),8)) //-->8
除去max(),min()
。还有一个clear()
的方法,顾名思义就是对传入的变量进行一个数据回收
go
func clear[T ~[]Type | ~map[Type]Type1](t T)
可以看到该方法主要服务于slice与map类型 对于切片,clear会将其数据变为默认值0,但是len,cap都不变
go
s:=[]int{1,2,3,4,5,6,7,8,9,10}
clear(s)
fmt.Printf("s:%v,len:%v,cap:%v",s,len(s),cap(s))
//s:[0 0 0 0 0 0 0 0 0 0],len:10,cap:10
而对于map,结果是这样的
go
m:=make(map[string]int,10)
m["a"]=1
fmt.Printf("m:%v,len:%v\n",m,len(m))
clear(m)
fmt.Printf("m:%v,len:%v",m,len(m))
//m:map[a:1],len:1
//m:map[],len:0
数据清空,桶也清空~
3.contenxt更新
context应该不必我多介绍了,Go语言特别常用的用于不同go协程之间的通信,收发取消信号、截至信号、元数据等上下文。
context这次更新主要是针对cancel()
发送取消信号增加了一个取消原因以及回调函数的增添
scss
func WithCancelCause(parent Context) (ctx Context, cancel CancelCauseFunc)
func WithTimeoutCause(parent Context, timeout time.Duration, cause error) (Context, CancelFunc)
func WithDeadlineCause(parent Context, d time.Time, cause error) (Context, CancelFunc)
如图在三个api中形参都增添了一个cause的错误类型,当cancel()调用或者超时后那么context.Cause(ctx)
方法就会获得我们自定义的错误,而非默认的ctx.Err()
中的context deadline exceeded 。这样做的好处显而易见,当我们程序出错的,根据我们自定义的错误去追踪错误显然会比默认的context deadline exceeded更精确
代码举例:
css
myctxErr:=fmt.Errorf("自定义ctxErr错误")
ctx,cancel:=context.WithTimeoutCause(context.Background(),1*time.Second,myctxErr)
defer cancel()
time.Sleep(2*time.Second)
fmt.Println(context.Cause(ctx))
//自定义ctxErr错误
而关于回调函数,
func AfterFunc(ctx Context, f func()) (stop func() bool)
其作用是在cancel取消或者超时后Evict
在一些场景下我们常常要用goroutine+select+channel 这套组合拳来实现监听取消的行为并为此做出后续的业务
而有了这个回调函数可以让我们用更简洁的代码来实现这一功能
举例:
go
func main(){
myctxErr:=fmt.Errorf("自定义ctxErr错误")
ctx,cancel:=context.WithTimeoutCause(context.Background(),5*time.Second,myctxErr)
context.AfterFunc(ctx,func1)
defer cancel()
time.Sleep(6*time.Second)
fmt.Println(context.Cause(ctx))
}
//evictEvent
func func1() {
fmt.Println(666)
}
//666
//自定义ctxErr错误
4.WASI支持
Go1.21 起,Go 将会支持 WASI 。预计先支持 WASI Preview1 标准,后续 WASI Preview2 成熟后会继续支持新标准。
而WASI全称为WebAssembly System Interface 不熟悉想了解的朋友可以观看WASI 前瞻 - 知乎 (zhihu.com)这篇文章
以上就是我分享所有内容,希望能够帮助到大家~~