Golang1.21更新内容全面介绍~

前言

在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也是基于泛型新增了maxmin函数,支持多种类型,多参数,在以前都需要自己去写这两个方法~

带大家看一下源码

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)这篇文章

以上就是我分享所有内容,希望能够帮助到大家~~

相关推荐
27669582929 小时前
boss直聘 __zp_stoken__ 逆向分析
java·python·node.js·go·boss·boss直聘·__zp_stoken__
绝无仅有3 天前
15个系统设计权衡关键点:构建高性能系统的黄金法则
面试·架构·go
绝无仅有3 天前
在 Go语言中一个字段可以包含多种类型的值的设计与接种解决方案
面试·架构·go
李歘歘4 天前
Golang——GPM调度器
java·开发语言·后端·golang·go·秋招·春招
Silber 甜4 天前
【GoLand】无法debug 无法运行
golang·go
白泽来了6 天前
Hugo|30分钟搭建完整的个人博客
开源·go
李歘歘8 天前
Golang笔记——切片与数组
开发语言·笔记·后端·golang·go
童先生9 天前
GoLand 如何集成 Netty?
go
卷心菜是俺10 天前
并发服务器框架——zinx
运维·服务器·go·github·代理模式
探索云原生10 天前
基于 Admission Webhook 实现 Pod DNSConfig 自动注入
云原生·kubernetes·go·dns