在go语言中自定义泛型的变长参数

本文涉及的产品
可观测监控 Prometheus 版,每月50GB免费额度
性能测试 PTS,5000VUM额度
Serverless 应用引擎免费试用套餐包,4320000 CU,有效期3个月
简介: 【7月更文挑战第8天】在Go语言中,由于官方1.18以前的版本不支持泛型,可以通过空接口和反射模拟泛型。泛型适用于通用数据结构和函数,虽牺牲了一些性能,但提高了代码复用和类型安全性。

1 简介

本文示例代码展示了一个自定义Container结构,利用反射创建指定类型切片并进行类型检查。
PutGet方法确保元素类型与容器匹配,否则返回错误。
代码在尝试添加不兼容类型或使用错误类型获取元素时会引发错误,体现了使用反射增强程序健壮性的特点。

2 自定义泛型:变长参数支持

在某些场景下,目前依然能使用反射来实现,比如泛型。
因为现在 Go 官方尚未在1.18版本之前的语法层面提供对泛型的支持,我们只能通过空接口结合反射来实现。

空接口 interface{} 本身可以表示任何类型的泛型,不过这个泛型太泛了,我们必须结合反射在运行时对实际传入的参数做类型检查,让其变得可控,

从而确保程序的健壮性,否则很容易因为传递进来的参数类型不合法导致程序崩溃。

下面我们通过一个自定义容器类型的实现来演示如何基于空接口和反射来实现泛型:

3 接口和容器的例子

基于 空接口和反射的 容器,实现泛型

     package main

    import (

        "fmt"

        "reflect"

    )

通过传入存储元素类型 和 容量 来初始化 容器

    type Container struct {

        s reflect.Value

    }

基于切片类型实现的容器,这里通过反射动态初始化这个底层切片

    func NewContainer(t reflect.Type, size int) *Container {

        return &Container{s: reflect.MakeSlice(reflect.SliceOf(t), 0, size)}

    }

通过反射对 实际传递来的 元素类型进行运行时检查。

如果与容器初始化设置的元素类型不同,则返回错误信息。

c.s.Type() 对应的是 切片类型,c.s.Type().Elem()对应的才是切片元素类型

    func (c *Container) Put(val interface{}) error {

    if reflect.ValueOf(val).Type() != c.s.Type().Elem() {

c.s 切片元素类型 与 传入参数不同

        return fmt.Errorf("put error:cannot put a %T into a slice of %s", c.s.Type().Elem())

       }
  • 如果类型检查通过则将其添加到容器

      c.s = reflect.Append(c.s, reflect.ValueOf(val))
    
           return nil
    
      }
    
      func (c *Container) Get(val interface{}) error {
    

还是通过反射对元素 类型进行检查,如果不通过则返回错误信息。

kind 与 Type 相比范围更大,表示类别,如指针,而Type则对应具体类型,如 *int

由于 val是指针类型,所有需要通过reflect.ValueOf(val).Elem() 获取指针指向的类型

    if reflect.ValueOf(val).Kind() != reflect.Ptr || reflect.ValueOf(val).Elem().Type() != c.s.Type().Elem() {

        return fmt.Errorf("get error:needs *%s but got %T", c.s.Type().Elem(), val)

      }

将容器第一个索引位置值赋值给 val 指针

    reflect.ValueOf(val).Elem().Set(c.s.Index(0))

然后删除容器第一个索引位置值

    c.s = c.s.Slice(1, c.s.Len()) 
    return nil 
    }

    func main() {

    nums := []int{1, 2, 3, 4, 5}

初始化容器,元素类型和nums中的元素类型相同

    c := NewContainer(reflect.TypeOf(nums[0]), 16)

    for _, n := range nums {

        if err := c.Put(n); err != nil {

            panic(err)

        }

从容器读取元素,将返回结果初始化为0

        num := 0

        if err := c.Get(&num); err != nil {

            panic(err)

        }

打印返回结果值

        fmt.Printf("%v, (%T)\n", num, num)

        }

        err := c.Put("s")     //put error:cannot put a *reflect.rtype into a slice of %!s(MISSING)

        err2 := c.Get("s100") //get error:needs *int but got string

        fmt.Println(err, err2)

    }
    ```

具体细节都已经在代码注释中详细标注了,执行上述代码,打印结果如下:

   -> chapter04 git:(main) x go run reflect/generic.go
   1(int)

如果我们试图添加其他类型元素到容器:

    ```
    if err := c.Put("s"); err != nil {    panic(err)}
    ```

或者存储返回结果的变量类型与容器内元素类型不符:

    ```
    if err := c.Get(num); err != nil {    panic(err)}
    ```

都会报错:

   -> generator git:(main) x go run reflect/generic.go
   panic: put error: can not put staring into a slice of int

   goroutine 1 [running]:
   main.main()
      ...
   exit status 2

  -> generator git:(main) x go run reflect/generic.go
   panic: get error: needs *int but got int

   goroutine 1 [running]:
   main.main()
      ...
   exit status 2

在这里,为了提高程序的健壮性,我们引入了错误处理机制,这块内容我们即将在下个章节中详细给大家介绍。

4 小结

  • 泛型的使用场景

    对特殊函数进行操作的函数

    使用反射时,可能更慢,它不是类型检查
    

    通用数据结构

    切片器数据结构,如链表 和 二叉树
    
    类型参数 替换 接口 存储数据,可避免类型断言
    

    不同的类型 需要实现一些通用方法时,有 通用方法时 可使用 类型

    许多函数都有类似代码,而不同点仅仅是 类型不同,此时使用 类型参数

    • 使用类型参数的场景的方式建议

         func ReadFour(r io.Reader) ([]byte, error)
      

不建议的方式

         func ReadFour[T io.Reader](r T) ([]byte, error)
目录
相关文章
|
2天前
|
Go
go语言的复数常量
【10月更文挑战第21天】
13 6
|
2天前
|
Go
go语言的浮点型常量
【10月更文挑战第21天】
9 4
|
2天前
|
编译器 Go
go语言的整型常量
【10月更文挑战第21天】
8 3
|
2天前
|
Serverless Go
Go语言中的并发编程:从入门到精通
本文将深入探讨Go语言中并发编程的核心概念和实践,包括goroutine、channel以及sync包等。通过实例演示如何利用这些工具实现高效的并发处理,同时避免常见的陷阱和错误。
|
3天前
|
安全 Go 开发者
代码之美:Go语言并发编程的优雅实现与案例分析
【10月更文挑战第28天】Go语言自2009年发布以来,凭借简洁的语法、高效的性能和原生的并发支持,赢得了众多开发者的青睐。本文通过两个案例,分别展示了如何使用goroutine和channel实现并发下载网页和构建并发Web服务器,深入探讨了Go语言并发编程的优雅实现。
10 2
|
网络协议 Go 存储
golang 自定义封包协议(转的)
package protocol import ( "bytes" "encoding/binary" ) const ( ConstHeader = "jackluo" ConstHeaderLength = 7 ...
2428 0
|
3天前
|
安全 网络协议 Go
Go语言网络编程
【10月更文挑战第28天】Go语言网络编程
89 65
|
3天前
|
网络协议 安全 Go
Go语言进行网络编程可以通过**使用TCP/IP协议栈、并发模型、HTTP协议等**方式
【10月更文挑战第28天】Go语言进行网络编程可以通过**使用TCP/IP协议栈、并发模型、HTTP协议等**方式
23 13
|
3天前
|
网络协议 安全 Go
Go语言的网络编程基础
【10月更文挑战第28天】Go语言的网络编程基础
17 8
|
3天前
|
Go
go语言编译时常量表达式
【10月更文挑战第20天】
11 3