Go语言for-range函数使用技巧实例探究

 更新时间:2024年01月21日 10:16:36   作者:晁岳攀(鸟窝) 鸟窝聊技术  
这篇文章主要为大家介绍了Go语言for-range函数使用技巧实例探究,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

Go range函数

Go 1.22 中可以 range 一个整数,比如下面的代码:

for i := range 10 {
    fmt.Println(i)
}

这个大家都已经知道了,其实对应的提案中还有一个隐藏的功能,就是可以 range 一个函数,比如下面的代码(摘自官方代码库internal/trace/v2/event.go[1]):

// Frames is an iterator over the frames in a Stack.
func (s Stack) Frames(yield func(f StackFrame) bool) bool {
 if s.id == 0 {
  return true
 }
 stk := s.table.stacks.mustGet(s.id)
 for _, f := range stk.frames {
  sf := StackFrame{
   PC:   f.pc,
   Func: s.table.strings.mustGet(f.funcID),
   File: s.table.strings.mustGet(f.fileID),
   Line: f.line,
  }
  if !yield(sf) {
   return false
  }
 }
 return true
}

就少有介绍了。

本文尝试介绍它,让读者先了解一下,它在 Go 1.22 中是一个实验性的功能,还不确定未来在哪个版本中会被正式支持。

官方 wiki 中也有一篇介绍: Rangefunc Experiment[2],类似问答的形式,也是必读的知识库。

这个功能去年 Russ Cox 发起讨论(#56413[3]), 并建立一个提案(#61405[4]),大家讨论都很激烈啊,几百次的讨论,所以我也不准备介绍前因后果了,直接了当的说结论。 先前, for-range所能遍历(迭代)的类型很有限,只能是 slice、数组、map、字符串、channel 等。 现在,除了上面的五种类型,还可以是整数和三种三种函数。

当然for x := range n { ... }等价于for x := T(0); x < n; x++ { ... }, 其中 T 是 n 的类型。这个大家都知道了。

range 的类型

三个函数可能大家不是很了解,很正常,目前这只是一个实验性的功能。当然 range 的类型如下:

Range 表达式第一个值第二个值
array or slice a [n]E, *[n]E, or []Eindex i inta[i] E
string s string typeindex i intsee below rune
map m map[K]Vkey k Km[k] V
channel c chan E, <-chan Eelement e E
integer n integer typeindex i int
function, 0 values f func(func()bool) bool
function, 1 value f func(func(V)bool) boolvalue v V
function, 2 values f func(func(K, V)bool) boolkey k Kv V

本文介绍的就是后三种形式

三种可遍历的函数

假设f是一个这样的函数:func(func()bool) bool, 那么for x := range f { ... }类似于f(func(x T1, y T2) bool { ... }),其中 for 循环移动到方法体中了。yield的 bool 返回值指示是否还要继续遍历。

对于这样一个f,下面的格式都可以:

for x, y := range f { ... }
for x, _ := range f { ... }
for _, y := range f { ... }
for x := range f { ... }
for range f { ... }

下面是一个例子:

 var fn = func(yield func(k int, v byte) bool) {
  for i := 0; i < 26; i++ {
   if !yield(i, byte('a'+i)) {
    return
   }
  }
 }

 for k, v := range fn {
  fmt.Printf("%d: %c\n", k, v)
 }

运行可以看到结果符合预期,我们遍历了 26 个小写字母,注意 range 的数据类型是我们的函数:

这里,fn 这个函数没有返回值,其实也可以有 bool 返回值,有 bool 返回值就可以组合多个 range 函数,可以容易写出复杂且难以维护的代码,减少自己失业的可能。 这里的yield函数接收两个参数,第一个是int类型,第二个是byte类型,返回值是bool类型,这个yield函数的返回值决定了是否继续遍历。当然这里我们可以写泛型的程序,这里为了简单,就不写了。

下面是一个ffunc(func(V)bool) bool的例子:

    var fn = func(yield func(v byte) bool) {
        for i := 0; i < 26; i++ {
            if !yield(byte('a' + i)) {
                return
            }
        }
    }

    for v := range fn {
        fmt.Printf("%c\n", v)
    }

当然 yield 函数也可以没有参数,比如func(func()bool) bool,下面这个例子就是无参数的形式,输出结果是 26。

package main

import "fmt"

func main() {
 var fn = func(yield func() bool) {
  for i := 0; i < 26; i++ {
   if !yield() {
    return
   }
  }
 }

 var count int
 for range fn {
  count++
 }
 fmt.Println(count)
}

如果不使用 for-range 函数的形式,我们可以进行改写,比如两个参数的列子:

 var fn = func(yield func(k int, v byte) bool) {
  for i := 0; i < 26; i++ {
   if !yield(i, byte('a'+i)) {
    return
   }
  }
 }

 fn(func(k int, v byte) bool {
  fmt.Printf("%d: %c\n", k, v)
  return true
 })

注意yield参数名称不是一个关键字,它只是一个普通的参数名称,可以随便取名字,但是为了模仿和其它语言中的generator,使用了yield这样一个名称,以至于代码更加易读。

看起来这个功能就是一个语法糖, 代码rangefunc/rewrite[5]将 range-over-func 代码写成非 range-over-func 代码的形式。

为什么要这样做?

标准库中就有archive/tar.Reader.Nextbufio.Reader.ReadBytebufio.Scanner.Scancontainer/ring.Ring.Dodatabase/sql.Rowsexpvar.Doflag.Visitgo/token.FileSet.Iteratepath/filepath.Walkgo/token.FileSet.Iterateruntime.Frames.Next 和sync.Map.Range等各种遍历的函数,所以如果有一种统一的格式更好。

第三方库中有更多的类似代码。

虽然这个功能还没有正式支持,但是我看到有些库摩拳擦掌准备使用了,而sqlrange[6]更进一步,已经支持了。

当然你使用它必须下载 Go 1.22 或者 gotip, 并且设置export GOEXPERIMENT=rangefunc

它提供了QueryExec可遍历函数。比如Query从一个表中查询Point数据:

type Point struct {
    X float64 `sql:"x"`
    Y float64 `sql:"y"`
}

for p, err := range sqlrange.Query[Point](db, `select x, y from points` "Point") {
    if err != nil {
        ...
    }
    ...
}

遍历查询和 ORM 一气呵成。这里的资源管理是自动的,底层的*sql.Rows遍历完会自动关闭。

参考文档

[1]internal/trace/v2/event.go: https://github.com/golang/go/blob/97daa6e94296980b4aa2dac93a938a5edd95ce93/src/internal/trace/v2/event.go#L262 

[2]Rangefunc Experiment: https://go.dev/wiki/RangefuncExperiment 

[3]#56413: https://github.com/golang/go/discussions/56413 

[4]#61405: https://github.com/golang/go/issues/61405 

[5]rangefunc/rewrite: https://go.googlesource.com/go/+/refs/changes/41/510541/7/src/cmd/compile/internal/rangefunc/rewrite.go 

[6]sqlrange: https://github.com/achille-roussel/sqlrange 

以上就是Go语言for-range函数使用技巧实例探究的详细内容,更多关于go for-range函数的资料请关注脚本之家其它相关文章!

相关文章

  • Golang库插件注册加载机制的问题

    Golang库插件注册加载机制的问题

    这篇文章主要介绍了Golang库插件注册加载机制,这里说的插件并不是指的golang原生的可以在buildmode中加载指定so文件的那种加载机制,需要的朋友可以参考下
    2022-03-03
  • GoLang中Strconv库有哪些常用方法

    GoLang中Strconv库有哪些常用方法

    这篇文章主要介绍了GoLang中Strconv库有哪些常用方法,strconv库实现了基本数据类型与其字符串表示的转换,主要有以下常用函数: Atoi()、Itia()、parse系列、format系列、append系列
    2023-01-01
  • golang中select语句的简单实例

    golang中select语句的简单实例

    Go的select语句是一种仅能用于channl发送和接收消息的专用语句,此语句运行期间是阻塞的,下面这篇文章主要给大家介绍了关于golang中select语句的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-06-06
  • 用go写的五子棋预测算法的实现

    用go写的五子棋预测算法的实现

    这篇文章主要介绍了用go写的五子棋预测算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • golang chan传递数据的性能开销详解

    golang chan传递数据的性能开销详解

    这篇文章主要为大家详细介绍了Golang中chan在接收和发送数据时因为“复制”而产生的开销,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下
    2024-01-01
  • go实现整型的二进制转化的方法

    go实现整型的二进制转化的方法

    这篇文章主要介绍了go实现整型的二进制转化的方法,本文给大家介绍的非常详细,具有一定的参考借鉴价值 ,需要的朋友可以参考下
    2019-07-07
  • Go语言开发k8s之ConfigMap操作解析

    Go语言开发k8s之ConfigMap操作解析

    这篇文章主要为大家介绍了Go语言开发k8s之ConfigMap操作示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-06-06
  • 一文详解golang通过io包进行文件读写

    一文详解golang通过io包进行文件读写

    这篇文章主要介绍了golang通过io包进行文件读写文中有详细的代码示例。对学习或工资有很好的帮助,需要的小伙伴可以参考阅读一下
    2023-04-04
  • 基于Golang container/list实现LRU缓存

    基于Golang container/list实现LRU缓存

    Least Recently Used (LRU) ,即逐出最早使用的缓存,这篇文章主要为大家介绍了如何基于Golang container/list实现LRU缓存,感兴趣的可以了解下
    2023-08-08
  • Go语言WaitGroup使用时需要注意的坑

    Go语言WaitGroup使用时需要注意的坑

    Go语言中WaitGroup的用途是它能够一直等到所有的goroutine执行完成,并且阻塞主线程的执行,直到所有的goroutine执行完成。之前一直使用也没有问题,但最近通过同事的一段代码引起了关于WaitGroup的注意,下面这篇文章就介绍了WaitGroup使用时需要注意的坑及填坑。
    2016-12-12

最新评论