将语法和接口作为参数进行操作

kaa*_*dar 14 syntax interface go

我是Go编程语言的新手,最近遇到了以下代码:

func (rec *ContactRecord) Less(other interface{}) bool {
  return rec.sortKey.Less(other.(*ContactRecord).sortKey);
}
Run Code Online (Sandbox Code Playgroud)

但是,我不明白函数签名背后的含义.它接受一个接口作为参数.你能解释一下这是怎么回事吗?谢谢

nem*_*emo 33

Go使用接口来进行类型的泛化.因此,如果您需要一个能够编写特定接口的函数

func MyFunction(t SomeInterface) {...}
Run Code Online (Sandbox Code Playgroud)

满足的每种类型SomeInterface都可以传递给MyFunction.

现在,SomeInterface可以看起来像这样:

type SomeInterface interface {
    SomeFunction()
}
Run Code Online (Sandbox Code Playgroud)

要满足SomeInterface,实现它的类型必须实现SomeFunction().

但是,如果您需要一个空接口(interface{}),则该对象不需要实现任何传递给该函数的方法:

func MyFunction(t interface{}) { ... }
Run Code Online (Sandbox Code Playgroud)

上面的这个函数将采用每种类型,因为所有类型都实现空接口.

获得类型

现在您可以拥有所有可能的类型,问题是如何获得之前实际放入的类型.空接口不提供任何方法,因此您无法在此类值上调用任何内容.

为此,您需要键入断言:让运行时检查值Y中是否存在类型X,如果是,则将其转换为该类型.

例:

func MyFunction(t interface{}) {
    v, ok := t.(SomeConcreteType)
    // ...
}
Run Code Online (Sandbox Code Playgroud)

在此示例中,输入参数t被声明为类型SomeConcreteType.如果t 实际上是类型SomeConcreteType,v将保留该类型的实例并且ok将为true.否则,ok将是假的.有关详细信息,请参阅类型断言的规范.


Jam*_*dge 6

一个interface变量可以保存任何类型的值,这些值为方法提供了来自接口声明的签名。由于interface{}没有指定任何方法,这样的变量可以存储任何类型的值。

然后该方法继续使用类型断言来检查它other实际上是一个*ContactRecord值(否则它会恐慌)。

然后您可能会问为什么不将方法声明为带*ContactRecord参数。最可能的原因是该*ContactRecord类型使用Less具有该签名的方法实现了某个接口。