首页 > 其他分享 >go泛型函数学习

go泛型函数学习

时间:2024-11-26 10:14:33浏览次数:4  
标签:型函数 int 学习 类型 Add func go type any

01 什么是泛型
泛型类似C++中的模板
Go 是一门强类型语言,意味着程序中的每个变量和值都有某种特定的类型,例如int、string 等。在函数签名中,我们需要对参数和返回值指定类型,如下所示:
func Add(a, b int) int
参数 a 和 b 的类型是 int,返回值类型也是 int,结果是 a 和 b 的和。
对两个 float64 求和的函数:
func AddFloat(a, b float64) float64
如果有更多其他的类型(比如字符串相加),可能需要写更多的对应版本函数,很不方便,也很繁琐,一堆复制粘贴的代码。

02 Go 中的泛型函数
使用泛型来实现上面的函数
func Add[T any](a, b T) T

Add 后面的 [T any],T 表示类型的标识,any 表示 T 可以是任意类型a、b 和返回值的类型 T 和前面的 T 是同一个类型为什么用 [],而不是其他语言中的 <>,官方有过解释,大概就是 <> 会有歧义。曾经计划使用 (),因为太容易混淆,最后使用了 []。
这样就表示,a、b 和返回值可以是任意类型,但它们的类型是同一个。那具体是什么类型如何确定呢?根据调用时的实际参数决定。因此,我们现在可以这么使用:
Add(1, 2)Add(2.1, 3.2)
不过,这时候代码会报错。

运行会报错:
type checking failed for mainprog.go2:8:9: invalid operation: operator + not defined for a (variable of type parameter type T)

03 约束
很显然,并非所有类型都支持加法操作。因此我们需要给出约束,指定可以进行加法操作的类型。
上面代码中,我们对类型 T 使用的是 any,相当于没有进行任何约束。现在我们给一个约束:
type Addable interface { type int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, uintptr, float32, float64, complex64, complex128, string}
这是新语法,叫做类型列表(type list)。
首先,Addable 重用了接口语法,即 interface 关键字,表示约束,具体约束的类型通过 type 指定,多个用逗号分隔。
现在 Add 函数中 T 的约束从 any 改为 Addable:
func Add[T Addable](a, b T) T { return a + b}
现在再次运行:https://gotipplay.golang.org/p/kR_B6OUyDXA,发现正常了。而且还支持字符串、复数等:
Add("polaris", "xu")
可见,约束可以是任意接口类型。(any 相当于空接口)
还有另外一种场景:可比较。比如 map 中的 key 要求是可比较的。比如下面的代码:
func findFunc[T any](a []T, v T) int { for i, e := range a { if e == v { return i } } return -1}
T 的约束是任意类型,而实际上并非所有类型都是可比较的。怎么办?我们当然可以向上面 Addable 一样定义一个约束,但为了方便,Go 内置提供了一个 comparable 约束,表示可比较的。参考下面代码:
package mainfunc findFunc[T comparable](a []T, v T) int { for i, e := range a { if e == v { return i } } return -1}func main() { print(findFunc([]int{1, 2, 3, 4, 5, 6}, 5))}

04 constraints 包
写泛型代码时,约束挺常见。再看一个例子,从切片中找出最大值:
func Max[T any](input []T) (max T) { for _, v := range input { if v > max { max = v } } return}
但运行会报错:
fmt.Println(Max([]int{1, 4, 2, 10}))// cannot compare v > max (operator > not defined for T)
这时,我们自然想到使用上面 Add 函数类似的办法,自定义一个约束:Ordered,把可能的类型都列上。
type Ordered interface { type int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, uintptr, float32, float64, string}
因为这样的需求挺常见的,为了方面,官方提供了一个新包:constraints,预定义了一些约束,具体查看:https://github.com/golang/go/issues/45458。
有了它,不需要自定义这个 Ordered 约束,而是使用 constraints 包中的,即:
func Max[T constraints.Ordered](input []T) (max T)

05 泛型类型
上面,我们介绍了泛型函数:即函数可以接受任意类型。注意和 interface{} 这样的任意类型区分开,泛型中的类型,在函数内部并不需要做任何类型断言和反射的工作,在编译期就可以确定具体的类型。
我们知道,Go 支持自定义类型,比如标准库 sort 包中的 IntSlice:
type IntSlice []int
此外,还有 StringSlice、Float64Slice 等,一堆重复代码。如果我们能够定义泛型类型,就不需要定义这么多不同的类型了。比如:
type Slice[T any] []T
能看懂吧。
在使用时,针对 int 类型,就是这样:
x := Slice[int]{1, 2, 3}
如果作为函数参数,这么使用:
func PrintSlice[T any](b Slice[T])
如果为这个类型定义方法,则是这样:
func (b Slice[T]) Print()
也就是说,Slice[T] 作为整体存在。
当然,泛型类型也可以做类型约束,而不是 any 类型:
type Slice[T comparable] []T

标签:型函数,int,学习,类型,Add,func,go,type,any
From: https://www.cnblogs.com/oldking1002/p/18569477

相关文章

  • gin.context学习
    gin.context是一个结构体类型,其定义如下:typeContextstruct{//定义了一些私有成员变量,用于存储请求和响应等信息writermemresponseWriterRequest*http.Request//保存request请求WriterResponseWriter//回写responseParamsParamshandlersHandlers......
  • Go进阶21:Goland 6条Tips让编程更有效率
    作为一个重度使用JetbrainsIDE全家桶的用户.这里我有几条可以加速您开发速度,减少您Paste/Copy敲打键盘的Tips.1.快速实现Interface操作步骤:光标移动到struct名称上Alt/Option+Enter选择ImplementInterface…Control+I搜索您需要实现的interface2.快速抽象I......
  • 鸿蒙学习全方位运维分析
    文章目录1、崩溃服务2、性能管理3、云服务监控4、故障监控和预防HUAWEIAppGalleryConnect提供低门槛、高效率、多场景的大数据能力,包括质量分析、性能调优、故障定位、行业风向等。同时支持多维度数据分析,智能诊断问题并给出解决方案,为开发者明确质量优化方向,提升......
  • 工作学习笔记(十五)Mybatis-Plus项目中使用eq
    在今天的工作中遇到了一个问题,在这记录一下第一次使用eq()。方法作用它的主要作用是在构建SQL查询语句的条件部分时,添加一个等于的判断条件。例如,当你想从数据库表中查询出某一字段值等于特定值的记录时,就可以使用eq()方法来实现这个条件构建。方法语法及参数说明语法:......
  • Java编程学习五
    一、数组的缺陷:二、集合框架三、Vector类四、ArrayList集合五、LinkedList集合六、泛型七、HashSet八、HashMap一、数组的缺陷:1.数组存在定容问题,一旦定义长度,就固定了容量,有时候定义的数据量不一定,很难保证容量不越出;如果需要存储更多或更少的元素,可能需要创建一个新......
  • Java编程学习六
    javaIO操作:Java中,Io操作主要是指使用Java进行输入和输出操作。Java中所有的IO机制都是基于数据流进行输入输出的。这些数据流表示了字符或者字节数据的流动序列。在Java中进行io操作,通常需要用到Java.io包中的类。比如说fiel类,用于表示文件和目录路径名的抽象表示形式。。......
  • Java学习笔记——2024.11.25
    2024.11.25一、Java_DOS原理1.DOS基本原理创建文件夹=>mdd:\\xxx消除文件夹=>rdd:\\xxx2.相对路径和绝对路径=>相对路径:从当前目录开始定位,形成的一个路径=>绝对路径:从顶级目录d,开始定位,形成的路径举例子:相对路径:..\..\abc2\test200\hello.txt......
  • python爬虫学习之--抓取汽车之家数据
    汽车之家的数据爬取还是比较简单的,遇到的坑如下:页面的页面编码格式:汽车之家的页面编码格式有三种,分别是**“GB2312”,“ISO-8859-1"和"UTF-8-SIG”,每次使用requests模块获取页面的html时,会随机出现其中的一种,其中页面编码格式为"GB2312",“ISO-8859-1”,可以正常显......
  • 【linux学习指南】初识Linux进程信号与使用
    文章目录......
  • 学习笔记(四十六):$$语法:内置组件双向同步
    概述:$$运算符为系统内置组件提供TS变量的引用,使得TS变量和系统内置组件的内部状态保持同步使用规则:1、当前$$支持基础类型变量,以及@State、@Link和@Prop装饰的变量2、$$绑定的变量变化时,会触发UI的同步刷新3、支持的组件 使用示例:@Entity@ComponentexportstructLog......