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

本文涉及的产品
应用实时监控服务-应用监控,每月50GB免费额度
云原生网关 MSE Higress,422元/月
应用实时监控服务-可观测链路OpenTelemetry版,每月50GB免费额度
简介: 【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)
目录
相关文章
|
17天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
18天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
92 71
|
17天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
100 67
|
18天前
|
存储 Go
go语言中映射
go语言中映射
32 11
|
9天前
|
Go 数据安全/隐私保护 UED
优化Go语言中的网络连接:设置代理超时参数
优化Go语言中的网络连接:设置代理超时参数
|
JavaScript 前端开发 Go
终于!Go 1.18 将支持泛型,来听听Go 核心技术团队 Russ Cox怎么说
终于!Go 1.18 将支持泛型,来听听Go 核心技术团队 Russ Cox怎么说
终于!Go 1.18 将支持泛型,来听听Go 核心技术团队 Russ Cox怎么说
|
18天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
27 7
|
20天前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
90 62
|
22天前
|
并行计算 安全 Go
Go语言中的并发编程:掌握goroutines和channels####
本文深入探讨了Go语言中并发编程的核心概念——goroutine和channel。不同于传统的线程模型,Go通过轻量级的goroutine和通信机制channel,实现了高效的并发处理。我们将从基础概念开始,逐步深入到实际应用案例,揭示如何在Go语言中优雅地实现并发控制和数据同步。 ####
|
20天前
|
Go
go语言for遍历映射(map)
go语言for遍历映射(map)
30 12