前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【译】golang 可变参数函数终极指南

【译】golang 可变参数函数终极指南

作者头像
goodspeed
发布于 2020-12-22 02:54:31
发布于 2020-12-22 02:54:31
3.5K00
代码可运行
举报
文章被收录于专栏:厉害了程序员厉害了程序员
运行总次数:0
代码可运行

使用常用模式学习关于go语言可变参数函数的一切

Ultimate Guide to Go Variadic Functions 原文地址 https://blog.learngoprogramming.com/golang-variadic-funcs-how-to-patterns-369408f19085

什么是可变参数函数?

可变参数函数是指传入参数是可变数量(0到更多)的函数。在输入的变量类型前面的省略号(三点)前缀即构成一个有效的变量。

2

声明一个可变参数名为“ names”,类型为string 的可变参数函数

一个简单的可变参数函数

这个 func 以字符串的形式返回传递的参数,字符串之间用空格分隔。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(names ...string) string {
  return strings.Join(names, " ")
}

你可以传入零个或多个参数

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
toFullname("carl", "sagan")// output: "carl sagan"
toFullname("carl")// output: "carl"
toFullname()// output: ""

什么时候使用可变参数函数?

  • 省略创建仅作为函数参数创建临时 slice 变量
  • 当输入参数的长度未知时
  • 表达你增加可读性的意图

例子:

看看 Go Stdlibfmt. Println 函数,就会明白它是如何使自己变得易于使用的。

它使用可变参数函数接受可选的输入参数数目。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func Println(a ...interface{})

如果它不是一个可变参数函数,它看起来会是这样的:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func Println(params []interface{})

你需要传递一个 slice 才能使用它ーー verbose,是的! :

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
fmt.Println([]interface{}{"hello", "world"})

相比而言,在可变参数函数形式中,它的使用是简单的:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
fmt.Println("hello", "world")
fmt.Println("hello")
fmt.Println()

在这之后的部分,将会介绍一些关于 可变参数函数 的细节和常用模式的例子。

切片与可变参数函数

可变参数在函数中会被转换为“新的”切片。可变参数实际上是 slice 类型的参数的语法糖。

3

不传入参数

如果不向其传递任何值,就相当于向可变参数函数传递了 nil 切片。

4

所有非空切片都有内建数组,而 nil 片则没有。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(names ...string) []string {
  return names
}

// names 内建数组为: nil

但是,当你向可变参数函数添加参数时,它将创建一个与你传入参数相关联的数组,而不再是一个空切片。

go语言内置函数append 将参数追加到现有的slice,并返回。append 也是一个可变参数函数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(names ...string) []string {
  return append(names, "hey", "what's up?")
}

toFullname()

// output: [hey what's up?]

如何传递一个切片?

通过将可变参数运算符... 加在现有切片后,可以将其传递给可变参数运算符。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
names := []string{"carl", "sagan")}

toFullname(names...)// output: "carl sagan"

这等同于以下调用:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
toFullname("carl", "sagan")

但是,有一点不同: ,在函数中将直接使用传入的切片而不是创建新的切片。关于这一点,请参阅以下内容。

5

像下面这样,你也可以将数组作为可变参数函数的参数:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
names := [2]string{"carl", "sagan"}
toFullname(names[:]...)

Passed slice’s spooky action at a distance

标题不知道怎么翻译...

假设你将一个 slice 作为参数传给一个可变参数函数:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
dennis := []string{"dennis", "ritchie"}
toFullname(dennis...)

再假设你修改了函数中变量参数的第一项:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(names ...string) string {
  names[0] = "guy"
  return strings.Join(names, " ")
}

修改它也会影响原始的切片。“ dennis”切片现在变成了:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
[]string{"guy", "ritchie"}

而不是原始值:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
[]string{"dennis", "ritchie"}

因为传入的 slice 与 func 内部的 slice 共享相同的底层数组,所以在 func 内部改变 slice 的值也会影响传入的 slice:

6

如果你直接传递参数(不使用切片) ,就不会发生这种情况。

动态传递多个切片

假设你想在 slice 传递给 func 之前,在 slice 前面添加“ mr. ”。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
names := []string{"carl", "sagan"}

首先append 函数会创建一个新的切片,然后将names展开, 然后将值依次添加到新创建的切片上,然后再将展开的结果传给 toFullname 函数:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
toFullname(append([]string{"mr."}, names...)...)

// output: "mr. carl sagan"

这和下面的代码一样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
names = append([]string{"mr."}, "carl", "sagan")
toFullname(names...)

// or with this:
toFullname([]string{"mr.", "carl", "sagan"}...)

// or with this—except passing an existing slice:
toFullname("mr.", "carl", "sagan")

返回传入的切片

不能使用可变参数作为返回结果类型,但是可以将其作为片返回。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func f(nums ...int) []int {
  nums[1] = 10
  return nums
}

当你将一个 slice 作为参数传入 f 时,它将返回一个相同的新 slice。传入和返回的切片是相关联的。两个中任意一个改变都会影响到另一个。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
nums  := []int{23, 45, 67}
nums2 := f(nums...)

这里,nums 和 nums2有相同的元素,因为它们都指向相同的底层数组。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
nums  = []int{10, 45, 67}
nums2 = []int{10, 45, 67}

? 这段代码 包含关于 slice 基础数组的详细说明

扩展操作符反模式

如果你有一个 funcs,它们唯一的用途就是接受可变数量的参数,那么最好使用可变参数函数代替使用 slice。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// Don't do this
toFullname([]string{"rob", "pike"}...)

// Do this
toFullname("rob", "pike")

运行代码

使用可变参数的长度

你可以使用可变参数的长度来改变函数的行为。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func ToIP(parts ...byte) string {
  parts = append(parts, make([]byte, 4-len(parts))...)  
  
  return fmt.Sprintf("%d.%d.%d.%d", 
    parts[0], parts[1], parts[2], parts[3])
}

ToIP func 将“ parts”作为可变参数,并使用 parts param 的长度返回默认值为0的字符串形式的 IP 地址。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
ToIP(255)   // 255.0.0.0
ToIP(10, 1) // 10.1.0.0
ToIP(127, 0, 0, 1) // 127.0.0.1

运行代码

✪ 变量函数的签名

尽管可变参数函数是一种语法糖,但它的签名类型标识与接受切片的函数是不同的。

举个例子看一下字符串切片和 ...字符串的区别是什么?

一个可变参数函数的签名:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func PrintVariadic(msgs ...string)// signature: func(msgs ...string)

非可变参数函数的签名:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func PrintSlice(msgs []string)// signature: func([]string)

它们的类型标识不一样,我们把它们赋值给变量:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
variadic := PrintVariadic   // variadic is a func(...string)
slicey := PrintSlice       // slice is a func([]string)

因此,其中一个不能替代另一个:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
slicey = variadic// error: type mismatch

运行代码

混合变量和非可变参数

你可以通过把非可变参数放在可变参数之前,将非可变参数与可变参数混合。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(id int, names ...string) string {
  return fmt.Sprintf("#%02d: %s", id, strings.Join(names, " "))
}

toFullname(1, "carl", "sagan")// output: "#01: carl sagan"

但是,你不能将非可变参数放到可变参数之后:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func toFullname(id int, names ...string, age int) string {}// error

运行代码

接受变量类型的参数

例如,Go Stdlib 的 Printf 函数,使用空接口类型接受任何类型的输入参数。你还可以使用空接口接受任意类型和任意数量的参数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func Printf(format string, a ...interface{}) (n int, err error) {
    /* this is a pass-through with a... */  
    return Fprintf(os.Stdout, format, a...)
}

fmt.Printf("%d %s %f", 1, "string", 3.14)    // output: "1 string 3.14"

为什么 Printf 只接受一个变量参数?

查看 Printf 的签名时,会发现它接受一个名为 format 的字符串和一个可变参数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func Printf(format string, a ...interface{})

这是因为format是必需的参数。Printf 强制您提供它,否则代码将无法编译。

如果它通过一个可变参数接收所有参数,那么调用方可能没有提供必要的格式化程序参数,或者从可读性角度来看,它不会像这个参数那么明确。它清楚地标明了 Printf 需要什么

此外,调用时不传入变量参数“a”,它将防止 Printf 在 函数中创建一个不必要的切片,一个值为 nil 切片。This may not be a clear win for Printf but it can be for you in your own code

你也可以在自己的代码中使用相同的模式。

注意空接口类型

interface{}类型也称为空接口类型,这意味着它绕过了自身的 Go 静态类型语义检查。不必要地使用它会给你带来弊大于利的后果。

例如,它可能强制你使用reflection,这是一个运行时特性(instead of fast and safe — compile-time)。你可能需要自己查找类型错误,而不是依赖于编译器帮你找到它们。

在使用空接口之前要仔细考虑,依靠显式类型和接口来实现所需的行为。

将切片传递给具有空接口的可变参数

你不能将一个普通的切片传递给一个具有空接口类型的可变参数。具体原因请阅读这里。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
hellos := []string{"hi", "hello", "merhaba"}

You expect this to work, but it doesn’t:

你可能期望这能生效,但事实并非如此:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
fmt.Println(hellos...)

因为,hello 是一个字符串,而不是一个空接口切片。可变参数或切片只能属于一种类型。

首先需要将 hellos slice 转换为一个空接口 slice:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
var ihellos []interface{} = make([]interface{}, len(hello))for i, hello := range hellos {
  ihellos[i] = hello
}

Now, the expansion operator will work:

现在,扩展运算符将开始生效:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
fmt.Println(ihellos...)
// output: [hi hello merhaba]

运行代码

函数式编程方面

你还可以使用可变参数函数接受可变数目的函数。让我们声明一个新的 formatter func 类型。格式化程序 func 获取并返回一个字符串:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
type formatter func(s string) string

让我们声明一个可变参数函数,它接受一个字符串和数量可选的可格式化的类型,以便使用一些pipeline来格式化字符串。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func format(s string, fmtrs ...formatter) string {
  for _, fmtr := range fmtrs {
    s = fmtr(s)
  }  return s
}

format(" alan turing ", trim, last, strings.ToUpper)// output: TURING

运行代码

您还可以使用channels、structs等来代替这种链式模式的函数。看* 这里 或者 这里 查看示例.

使用结果为slice的函数作为可变参数

让我们重复使用上面的“format func”来创建一个可重用的格式化管道构建器:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func build(f string) []formatter {
  switch f {
    case "lastUpper":
      return []formatter{trim, last, strings.ToUpper}
    case "trimUpper":
      return []formatter{trim, strings.ToUpper}
    // ...etc
    default:
      return identityFormatter
  }
}

然后使用 expand 运算符运行它,最后将结果提供给格式 func:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
format(" alan turing ", build("lastUpper")...)// output: TURING

可变参数选项模式

You may have already been familiar with this pattern from other OOP langs and this has been re-popularized again in Go by Rob Pike here back in 2014. It’s like the visitor pattern.

你可能已经熟悉这种来自其他 OOP 语言的模式,这种模式在2014年 Rob Pike 的 Go 中再次流行起来。这就像是访客模式。

这个例子对你来说可能有点难。如果有不理解的请及时提问<作者不在,查看原文链接提问吧?>。

让我们创建一个 Logger,可以使用可选模式在运行时更改详细程度和前缀:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
type Logger struct {
  verbosity
  prefix string
}

使用一个可变的选项参数来改变logger的行为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func (lo *Logger) SetOptions(opts ...option) {
  for _, applyOptTo := range opts {
    applyOptTo(lo)
  }
}

我们创建一些返回配置方法的函数,它们在一个闭包中改变 Logger 的操作行为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func HighVerbosity() option {
  return func(lo *Logger) {
    lo.verbosity = High
  }
}

func Prefix(s string) option {
  return func(lo *Logger) {
    lo.prefix = s
  }
}

现在,让我们用默认选项创建一个新的 Logger:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
logger := &Logger{}

然后通过变量参数向记录器提供选项:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
logger.SetOptions(
  HighVerbosity(), 
  Prefix("ZOMBIE CONTROL"),
)

现在让我们检查一下输出:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
logger.Critical("zombie outbreak!")
// [ZOMBIE CONTROL] CRITICAL: zombie outbreak!

logger.Info("1 second passed")
// [ZOMBIE CONTROL] INFO: 1 second passed

运行代码

✪ 无穷无尽的精神食粮!

  • 在 Go 2中,有一些可变函数的行为的计划*这里, here 这里,及这里.
  • 你可以在 Go 语言标准文档里找到更正式的可变参数函数指南,这里, 这里, 这里 及这里.
  • 使用来自 c 的可变函数.
  • 你可以找到很多语言的可变参数函数声明这里.自由探索吧

好了,就到这了。谢谢你们的阅读。

参考链接

  • [1] Ultimate Guide to Go Variadic Functions 原文地址: https://blog.learngoprogramming.com/golang-variadic-funcs-how-to-patterns-369408f19085
  • [2] 类型标识: https://golang.org/ref/spec#Type_identity
  • [3] Go Stdlib 的 Printf: https://golang.org/src/fmt/print.go#L189
  • [4] reflection: https://blog.golang.org/laws-of-reflection
  • [5] 请阅读这里: https://golang.org/doc/faq#convert_slice_of_interface
  • [6] MultiReader: https://golang.org/pkg/io/#MultiReader
  • [7] text/template/parse: https://golang.org/src/text/template/parse/parse.go?s=1642:1753#L41
  • [8] visitor pattern: https://en.wikipedia.org/wiki/Visitor_pattern
  • `[9] 模式: https://commandcenter.blogspot.com.tr/2014/01/self-referential-functions-and-design.html
  • `[10]访客模式: https://en.wikipedia.org/wiki/Visitor_pattern
  • [11]https://github.com/golang/go/issues/15209
  • [12]https://github.com/golang/go/issues/18605: https://github.com/golang/go/issues/18605
  • `[13]https://github.com/golang/go/issues/19218*: https://github.com/golang/go/issues/19218
  • [14] https://golang.org/ref/spec#Passing_arguments_to_..._parameters
  • [15] https://golang.org/ref/spec#Appending_and_copying_slices
  • [16] https://golang.org/ref/spec#Type_identity
  • [17] 使用来自 c 的可变函数: https://sunzenshen.github.io/tutorials/2015/05/09/cgotchas-intro.html
  • [18] Variadic_function: https://rosettacode.org/wiki/Variadic_function
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-08-16,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 四月 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Go程序例子(13):可变参数函数
可变参数函数(Variadic functions)是 Go 中一种允许接受任意数量参数的函数。fmt.Println 就是一个常见的可变参数函数的例子。
用户11078756
2024/12/11
830
Go程序例子(13):可变参数函数
[日常] Go语言圣经-可变参数习题
1.参数数量可变的函数称为为可变参数函数,例子就是fmt.Printf和类似函数 2.参数列表的最后一个参数类型之前加上省略符号“...” 3.虽然在可变参数函数内部,...int 型参数的行为看起来很像切片类型,但实际上,可变参数函数和以切片作为参数的函数是不同的 类型不同:fmt.Printf("%T\n", f) 4.函数名的后缀f是一种通用的命名规范,代表该可变参数函数可以接收Printf风格的格式化字符串 5.interfac{}表示函数的最后一个参数可以接收任意类型
唯一Chat
2019/09/10
6030
[日常] Go语言圣经-可变参数习题
2.GO-可变参数函数,匿名函数和函数变量
2.1.可变参数函数 可变参数指参数的个数可以是任意个 可变参数必须在参数列表最后的位置,在参数名和类型之间添加三个点表示可变参数函数 声明函数时,在函数体把可变参数当作切片使用即可 package main import "fmt" func demo(name string, hover ... string) { fmt.Println(name,"的爱好是") for i,n := range hover{ fmt.Println(i,n) } } func main() {
zhang_derek
2019/08/06
8530
Golang可变参数
既然我们知道了可变参数会被转换成切片,那么通过go语法糖,可以将一个存在的分配当作可变参数的参数。
春哥大魔王
2018/09/21
8280
Go语言的可变(不定)长参数函数
上面的三个点(…)表示args是一个可变参数。在函数Function中,参数args会被当做一个slice来处理的。
sean.liu
2022/09/28
1K0
可变参数的函数
C语言允许定义参数数量可变的函数,这称为可变参数函数(variadic function)。这种函数需要固定数量的强制参数,后面是数量可变的可选参数。其中,强制参数必须至少一个,可选参数数量可变,类型可变,可选参数的数量由强制参数的值决定。 C 语言中最常用的可变参数函数例子是 printf()和 scanf()。这两个函数都有一个强制参数,即格式化字符串。格式化字符串中的转换修饰符决定了可选参数的数量和类型。 可变参数函数格式:int fun(int a,...)
用户7272142
2023/10/11
4320
C/C++开发基础——可变参数与可变参数模板
1.如果可变参数的参数类型相同,可以使用标准库中的initializer_list。
Coder-ZZ
2023/09/04
8520
C/C++开发基础——可变参数与可变参数模板
C++核心准则ES.34:不要定义C风格的可变参数函数
Not type safe. Requires messy cast-and-macro-laden code to get working right.
面向对象思考
2020/05/20
9650
由phithon的一个题目谈可变参数函数
可变参数函数是指参数个数可变的函数,在函数声明和定义的时候并没有明确的指出函数需要的参数个数,具体有多少个参数,是在调用的时候确定的. 可变参数函数并不是什么新奇的东西,早在我们学c语言的时候,就见过,例如我们常用的printf()和scanf()函数. printf() 的函数原型是
用户1879329
2023/02/27
1.3K0
C语言“…”占位符及可变参数函数
C语言函数的参数传递总是固定了个数,那么有没有传递任意个数参数的方法呢?在C++中,函数重载提供了多种参数传递的解决办法,但也不是任意参数个数。事实上,C语言是提供任意数量参数的解决方案的。
sean.liu
2022/08/03
1.3K0
Go语言之可变参数函数
1.不输入参数;2.输入一个参数;3.输入多个参数;4.按照slice的方式输入。
灰子学技术
2023/10/30
2410
Go语言之可变参数函数
Go 函数可变参数传参
众所周知,Go语言是严格类型语言,而开发的时候又遇到传入参数不定的情况,怎么办?golang 为我们提供了接入多值参数用于解决这个问题。
IT工作者
2022/06/30
1.8K0
Go 三个点(...)用法
s如果使用s...符号解压缩切片,则可以将切片直接传递给可变参数函数。在这种情况下,不会创建新的切片。
孤烟
2020/09/27
3.2K0
给可变参数函数传入切片 转
有一个可以直接将切片传入可变参数函数的语法糖,你可以在在切片后加上 ... 后缀。如果这样做,切片将直接传入函数,不再创建新的切片
双面人
2019/04/10
1.1K0
《Go语言程序设计》读书笔记(二)函数
《Go 语言程序设计》在线阅读地址:https://yar999.gitbooks.io/gopl-zh/content/
KevinYan
2019/12/30
4660
python中函数的可变参数
print(1,2,"hello","刘金玉编程","编程创造城市",end="$$$")
刘金玉编程
2019/07/30
2.5K0
可变参数(c/c++)
有时候我们在编写函数时,可能不知道要传入的参数个数,类型 。比如我们要实现一个叠加函数,再比如c语言中的printf,c++中的emplace_last()。
薄荷冰
2024/02/17
1.1K0
可变参数(c/c++)
Golang动态可变函数参数 参数默认值
作者:matrix 被围观: 4 次 发布时间:2024-08-17 分类:Golang | 无评论 »
HHTjim 部落格
2024/08/17
2060
举例分析可变参数函数实现的过程
函数栈调用 对于C语言,其调用遵循_cdecl规则: 1.所有参数从右到左依次入栈。 2.这些参数由调用者清除,称为手动清除。 3.被调用函数不会要求调用者传递多少参数,调用者传递过多或者过少的参数,甚至完全不同的参数都不会产生编译阶段的错误。(简化的将就是调用参数的类型和数量不会产生编译阶段的错误)
lexingsen
2022/02/24
6090
举例分析可变参数函数实现的过程
Golang语言--可变参数函数,何时该使用省略号(...)
今天的一个例子中发现,对于在调用可变参数函数时,不是总能使用省略号将一个切片展开,有时候编译器可能会报错,为了清除的说明这个问题,我用几个小例子一步一步说明。 1、提出假想的需求 假如想要在一堆数据
李海彬
2018/03/21
2K0
Golang语言--可变参数函数,何时该使用省略号(...)
相关推荐
Go程序例子(13):可变参数函数
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验