【问题标题】:idiom for mapping multiple "arguments" to single variadic parameter将多个“参数”映射到单个可变参数的习语
【发布时间】:2019-07-08 13:48:17
【问题描述】:

有没有办法在 golang 中采用可变参数的函数中使用 splatted 参数和简单参数(定义如下)的组合?如果没有,是否有一个近似此功能的著名成语?如果可能,是否有不需要重复类型名称的近似此功能的成语?


假设我有一个带有...T 类型的 golang 变量参数的函数。 您在通话现场的选择似乎仅限于:

  • 多个简单参数,每个类型为T,即f(…… x1, x2, x3 ……)
  • []T 类型的单个 splatted 参数,即 f(…… ...xs ……)

在 Python 等其他一些语言中,可以捕获简单参数和带有可变参数的 splatted 参数的组合:

$ python
>>> def foo(*args): return args
... 
>>> foo(1, 2, *[3, 4])
(1, 2, 3, 4)

但是,同样的事情在 go 中似乎不起作用。

// splat.go
package main

import "os"
import "fmt"

func write(args ...string) {
        for _, item := range args {
                fmt.Printf("%s\n", item)
        }
}

func main() {
        write("foobarbaz", os.Args[1:]...)
}

从错误消息中可以清楚地看出,splatted 参数和简单参数没有映射到同一个参数。

$ go run splat.go 
# command-line-arguments
./splat.go:14: too many arguments in call to write

【问题讨论】:

    标签: go


    【解决方案1】:

    创建一个包含所有参数的切片并将该切片分解。

    write(append([]string{"foobarbaz"}, os.Args[1:]...)...)
    

    如果string, []string 模式在应用程序代码中很常见,那么考虑编写一个函数来简化调用站点的代码:

    func stringArgs(s string, args []string) []string { 
       return append([]string{s}, args...)
    }
    
    ...
    
    write(stringArgs("foobarbaz", os.Args[1:])...)
    

    func writeStringArgs(s string, args []string) {
        write(append([]string{"foobarbaz"}, os.Args[1:]...)...)
    }
    
    ...
    
    writeStringArgs(s, os.Args[1:])
    

    【讨论】:

    • 有没有不需要重复类型名称([]string{……})的方法?
    • interface{} 是 go 中的包罗万象的泛型类型 - 但只有在目标函数明确接受此泛型类型时才会起作用。我认为您正在寻找泛型 - go 1.X 不支持。 go 2.X 可能支持泛型,但那还有很长的路要走。
    • @colminator ... go 没有一流的或用户可定义的泛型,但它确实具有某些内置的东西,例如rangeappend,它们的行为类似于多态函数。如果不命名相关类型,似乎不可能在切片之前添加。
    • 只要根类型是interface{},你当然可以实现你想要的。请参阅反射包。但重申我之前的评论,您的目标函数必须接受接口{}类型才能实现任何通用行为。
    • @GregoryNisbet 如果不提​​及类型,就无法创建切片。
    猜你喜欢
    • 2019-11-26
    • 2023-03-24
    • 2017-08-15
    • 1970-01-01
    • 2012-08-28
    • 1970-01-01
    • 1970-01-01
    • 2020-02-01
    • 1970-01-01
    相关资源
    最近更新 更多