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

本文涉及的产品
Serverless 应用引擎 SAE,800核*时 1600GiB*时
容器服务 Serverless 版 ACK Serverless,317元额度 多规格
性能测试 PTS,5000VUM额度
简介: 【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)
目录
相关文章
|
7天前
|
Go
Go 语言循环语句
在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句。
17 1
|
6天前
|
Go
Go to Learn Go之命令行参数
Go to Learn Go之命令行参数
18 8
|
7天前
|
Go 开发者
探索Go语言的并发之美
在Go语言的世界里,"并发"不仅仅是一个特性,它是一种哲学。本文将带你领略Go语言中goroutine和channel的魔力,揭示如何通过Go的并发机制来构建高效、可靠的系统。我们将通过一个简单的示例,展示如何利用Go的并发特性来解决实际问题,让你的程序像Go一样,轻盈而强大。
|
8天前
|
JSON Go API
使用Go语言和Gin框架构建RESTful API:GET与POST请求示例
使用Go语言和Gin框架构建RESTful API:GET与POST请求示例
|
8天前
|
Go
go语言创建字典
go语言创建字典
|
7天前
|
Go
Go 语言接口
Go 语言提供了另外一种数据类型即接口,它把所有的具有共性的方法定义在一起,任何其他类型只要实现了这些方法就是实现了这个接口。 接口可以让我们将不同的类型绑定到一组公共的方法上,从而实现多态和灵活的设计。
|
前端开发 rax Linux
go语言如何使用rbp, rsp, 参数如何传递, 为什么go的返回值写在后面
# 为什么go的返回值写在后面 go一直被鼓吹语法比java好, 性能跟c一样. 让我们来看一看go语言各部分对应的二进制指令, 是如何实现的 现在的想法是写个一系列文章, 把go的所有语法的实现方式都分析一遍, 不知道会不会半途而废 ### 本文所有的分析方法, 结论都是本人猜测的, 查各种文档太费时间了, 当然不是乱猜, 都是有依据的 先看栈回溯最基本的方法, rbp, r
1712 0
|
9天前
|
安全 Go 数据处理
探索Go语言的并发之美:Goroutines与Channels
在Go语言的世界里,"并发"不仅仅是一个概念,它是一种生活的方式。本文将带你领略Go语言中Goroutines和Channels的魔力,它们是如何让并发编程变得既简单又高效。我们将通过一个简单的示例,展示如何使用这些工具来构建一个高性能的网络服务。
|
9天前
|
关系型数据库 Go 数据处理
高效数据迁移:使用Go语言优化ETL流程
在本文中,我们将探索Go语言在处理大规模数据迁移任务中的独特优势,以及如何通过Go语言的并发特性来优化数据提取、转换和加载(ETL)流程。不同于其他摘要,本文不仅展示了Go语言在ETL过程中的应用,还提供了实用的代码示例和性能对比分析。
|
9天前
|
Go 定位技术 索引
Go 语言Map(集合) | 19
Go 语言Map(集合) | 19
下一篇
无影云桌面