泛型
泛型
lycheeKingGo 1.18版本增加了对泛型的支持,泛型也是自 Go 语言开源以来所做的最大改变。
什么是泛型
泛型允许程序员在强类型程序设计语言中编写代码时使用一些以后才指定的类型,在实例化时作为参数指明这些类型。ーー换句话说,在编写某些代码或数据结构时先不提供值的类型,而是之后再提供。
泛型是一种独立于所使用的特定类型的编写代码的方法。使用泛型可以编写出适用于一组类型中的任何一种的函数和类型。
为什么需要泛型
假设我们需要实现一个反转切片的函数——reverse
。
1 | func reverse(s []int) []int { |
可是这个函数只能接收[]int
类型的参数,如果我们想支持[]float64
类型的参数,我们就需要再定义一个reverseFloat64Slice
函数。
1 | func reverseFloat64Slice(s []float64) []float64 { |
如果要想支持[]string
类型切片就要定义reverseStringSlice
函数,如果想支持[]xxx
就需要定义一个reverseXxxSlice
…
一遍一遍地编写相同的功能是低效的,实际上这个反转切片的函数并不需要知道切片中元素的类型,但为了适用不同的类型我们把一段代码重复了很多遍。
Go1.18之前我们可以尝试使用反射去解决上述问题,但是使用反射在运行期间获取变量类型会降低代码的执行效率并且失去编译期的类型检查,同时大量的反射代码也会让程序变得晦涩难懂。
类似这样的场景就非常适合使用泛型。从Go1.18开始,使用泛型就能够编写出适用所有元素类型的“普适版”reverse
函数。
1 | func reverseWithGenerics[T any](s []T) []T { |
泛型语法
泛型为Go语言添加了三个新的重要特性:
- 函数和类型的类型参数。
- 将接口类型定义为类型集,包括没有方法的类型。
- 类型推断,它允许在调用函数时在许多情况下省略类型参数。
类型参数
类型形参和类型实参
我们之前已经知道函数定义时可以指定形参,函数调用时需要传入实参。
现在,Go语言中的函数和类型支持添加类型参数。类型参数列表看起来像普通的参数列表,只不过它使用方括号([]
)而不是圆括号(()
)。
借助泛型,我们可以声明一个适用于一组类型的min
函数。
1 | func min[T int | float64](a, b T) T { |
类型实例化
这次定义的min
函数就同时支持int
和float64
两种类型,也就是说当调用min
函数时,我们既可以传入int
类型的参数。
1 | m1 := min[int](1, 2) // 1 |
也可以传入float64
类型的参数。
1 | m2 := min[float64](-0.1, -0.2) // -0.2 |
向 min
函数提供类型参数(在本例中为int
和float64
)称为实例化( instantiation )。
1 | m2 := min[float64](-0.1, -0.2) // -0.2 |
向 min
函数提供类型参数(在本例中为int
和float64
)称为实例化( instantiation )。
类型实例化分两步进行:
- 首先,编译器在整个泛型函数或类型中将所有类型形参(type parameters)替换为它们各自的类型实参(type arguments)。
- 其次,编译器验证每个类型参数是否满足相应的约束。
在成功实例化之后,我们将得到一个非泛型函数,它可以像任何其他函数一样被调用。例如:
1 | fmin := min[float64] // 类型实例化,编译器生成T=float64的min函数 |
min[float64]
得到的是类似我们之前定义的minFloat64
函数——fmin
,我们可以在函数调用中使用它。
类型参数的使用
除了函数中支持使用类型参数列表外,类型也可以使用类型参数列表。
1 | type Slice[T int | string] []T |
在上述泛型类型中,T
、K
、V
都属于类型形参,类型形参后面是类型约束,类型实参需要满足对应的类型约束。
泛型类型可以有方法,例如为上面的Tree
实现一个查找元素的Lookup
方法。
1 | func (t *Tree[T]) Lookup(x T) *Tree[T] { ... } |
要使用泛型类型,必须进行实例化。Tree[string]
是使用类型实参string
实例化 Tree
的示例。
1 | var stringTree Tree[string] |
类型约束
普通函数中的每个参数都有一个类型; 该类型定义一系列值的集合。例如,我们上面定义的非泛型函数minFloat64
那样,声明了参数的类型为float64
,那么在函数调用时允许传入的实际参数就必须是可以用float64
类型表示的浮点数值。
类似于参数列表中每个参数都有对应的参数类型,类型参数列表中每个类型参数都有一个类型约束。类型约束定义了一个类型集——只有在这个类型集中的类型才能用作类型实参。
Go语言中的类型约束是接口类型。
就以上面提到的min
函数为例,我们来看一下类型约束常见的两种方式。
类型约束接口可以直接在类型参数列表中使用。
1 | // 类型约束字面量,通常外层interface{}可省略 |
作为类型约束使用的接口类型可以事先定义并支持复用。
1 | // 事先定义好的类型约束类型 |
在使用类型约束时,如果省略了外层的interface{}
会引起歧义,那么就不能省略。例如:
1 | type IntPtrSlice [T *int] []T // T*int ? |
类型集
Go1.18开始接口类型的定义也发生了改变,由过去的接口类型定义方法集(method set)变成了接口类型定义类型集(type set)。也就是说,接口类型现在可以用作值的类型,也可以用作类型约束。
把接口类型当做类型集相较于方法集有一个优势: 我们可以显式地向集合添加类型,从而以新的方式控制类型集。
Go语言扩展了接口类型的语法,让我们能够向接口中添加类型。例如
1 | type V interface { |
上面的代码就定义了一个包含 int
、 string
和 bool
类型的类型集。
在 Go 1.18中,接口既可以像以前一样包含方法和嵌入式接口,但也可以嵌入非接口类型、类型并集和基础类型的集合。
当用作类型约束时,由接口定义的类型集精确地指定允许作为相应类型参数的类型。
|
符号T1 | T2
表示类型约束为T1和T2这两个类型的并集,例如下面的Integer
类型表示由Signed
和Unsigned
组成。1
2
3type Integer interface {
Signed | Unsigned
}~
符号~T
表示所以底层类型是T的类型,例如~string
表示所有底层类型是string
的类型集合。1
type MyString string // MyString的底层类型是string
注意:
~
符号后面只能是基本类型。
接口作为类型集是一种强大的新机制,是使类型约束能够生效的关键。目前,使用新语法表的接口只能用作类型约束。
any
空接口在类型参数列表中很常见,在Go 1.18引入了一个新的预声明标识符,作为空接口类型的别名。
1 | // src/builtin/builtin.go |
由此,我们可以使用如下代码:
1 | func foo[S ~[]E, E any]() { |
constrains
https://pkg.go.dev/golang.org/x/exp/constraints 包提供了一些常用类型。
类型推断
最后一个新的主要语言特征是类型推断。从某些方面来说,这是语言中最复杂的变化,但它很重要,因为它能让人们在编写调用泛型函数的代码时更自然。
函数参数类型推断
对于类型参数,需要传递类型参数,这可能导致代码冗长。回到我们通用的 min
函数:
1 | func min[T int | float64](a, b T) T { |
类型形参T
用于指定a
和b
的类型。我们可以使用显式类型实参调用它:
1 | var a, b, m float64 |
在许多情况下,编译器可以从普通参数推断 T
的类型实参。这使得代码更短,同时保持清晰。
1 | var a, b, m float64 |
这种从实参的类型推断出函数的类型实参的推断称为函数实参类型推断。函数实参类型推断只适用于函数参数中使用的类型参数,而不适用于仅在函数结果中或仅在函数体中使用的类型参数。例如,它不适用于像 MakeT [ T any ]() T
这样的函数,因为它只使用 T
表示结果。
约束类型推断
Go 语言支持另一种类型推断,即约束类型推断。接下来我们从下面这个缩放整数的例子开始:
1 | // Scale 返回切片中每个元素都乘c的副本切片 |
这是一个泛型函数适用于任何整数类型的切片。
现在假设我们有一个多维坐标的 Point
类型,其中每个 Point
只是一个给出点坐标的整数列表。这种类型通常会实现一些业务方法,这里假设它有一个String
方法。
1 | type Point []int32 |
由于一个Point
其实就是一个整数切片,我们可以使用前面编写的Scale
函数:
1 | func ScaleAndPrint(p Point) { |
不幸的是,这代码会编译失败,输出r.String undefined (type []int32 has no field or method String
的错误。
问题是Scale
函数返回类型为[]E
的值,其中E
是参数切片的元素类型。当我们使用Point
类型的值调用Scale
(其基础类型为[]int32)时,我们返回的是[]int32
类型的值,而不是Point
类型。这源于泛型代码的编写方式,但这不是我们想要的。
为了解决这个问题,我们必须更改 Scale
函数,以便为切片类型使用类型参数。
1 | func Scale[S ~[]E, E constraints.Integer](s S, c E) S { |
我们引入了一个新的类型参数
S
,它是切片参数的类型。我们对它进行了约束,使得基础类型是S
而不是[]E
,函数返回的结果类型现在是S
。由于E
被约束为整数,因此效果与之前相同:第一个参数必须是某个整数类型的切片。对函数体的唯一更改是,现在我们在调用make
时传递S
,而不是[]E
。
现在这个Scale
函数,不仅支持传入普通整数切片参数,也支持传入Point
类型参数。
这里需要思考的是,为什么不传递显式类型参数就可以写入 Scale
调用?也就是说,为什么我们可以写 Scale(p, 2)
,没有类型参数,而不是必须写 Scale[Point, int32](p, 2)
?
新
Scale
函数有两个类型参数——S
和E
。在不传递任何类型参数的Scale(p, 2)
调用中,如上所述,函数参数类型推断让编译器推断S
的类型参数是Point
。但是这个函数也有一个类型参数E
,它是乘法因子c
的类型。相应的函数参数是2
,因为2
是一个非类型化的常量,函数参数类型推断不能推断出E
的正确类型(最好的情况是它可以推断出2
的默认类型是int
,而这是错误的,因为Point 的基础类型是[]int32
)。相反,编译器推断E
的类型参数是切片的元素类型的过程称为约束类型推断。
约束类型推断从类型参数约束推导类型参数。当一个类型参数具有根据另一个类型参数定义的约束时使用。当其中一个类型参数的类型参数已知时,约束用于推断另一个类型参数的类型参数。
通常的情况是,当一个约束对某种类型使用 ~type 形式时,该类型是使用其他类型参数编写的。我们在 Scale
的例子中看到了这一点。S
是 ~[]E
,后面跟着一个用另一个类型参数写的类型[]E
。如果我们知道了 S
的类型实参,我们就可以推断出E
的类型实参。S
是一个切片类型,而 E
是该切片的元素类型。
总结
总之,如果你发现自己多次编写完全相同的代码,而这些代码之间的唯一区别就是使用的类型不同,这个时候你就应该考虑是否可以使用类型参数。
泛型和接口类型之间并不是替代关系,而是相辅相成的关系。泛型的引入是为了配合接口的使用,让我们能够编写更加类型安全的Go代码,并能有效地减少重复代码。