Golang泛型与反射的应用详解

 更新时间:2022-06-15 10:09:28   作者:佚名   我要评论(0)

目录1. 泛型1.1 定义1.2 例子1.3 自定义泛型类型1.4 泛型与switch结合使用1.5 泛型实战2. 反射2.1 定义2.2 方法2.3 反射读取2.4 反射操作2.5

1. 泛型

1.1 定义

  • 泛型生命周期只在编译期,旨在为程序员生成代码,减少重复代码的编写
  • 在比较两个数的大小时,没有泛型的时候,仅仅只是传入类型不一样,我们就要再写一份一模一样的函数,如果有了泛型就可以减少这类代码

1.2 例子

// SumInts 将map的值相加,如果需要添加的数据类型不同,那么就需要定义两个
func SumInts(m map[string]int64) int64 {
    var s int64
    for _, v := range m {
        s += v
    }
    return s
}
func SumFloats(m map[string]float64) float64 {
    var s float64
    for _, v := range m {
        s += v
    }
    return s
}

如果使用泛型的话只需要定义泛型方法即可(如果报一下编译错误的话,是idea版本过低,升级版本即可,但是运行没有问题)

func main() {
	ints := make(map[string]int64, 5)
	ints["name"] = 5
	ints["value"] = 6
	floats := make(map[string]float64, 5)
	floats["name"] = 5.6
	floats["value"] = 6.5
	fmt.Printf("Gnneric sums: %v and %v\n", 
		SumIntsOrFloats[string, int64](ints), 
		SumIntsOrFloats[string, float64](floats))
    //可以将类型删除
    fmt.Printf("Gnneric sums: %v and %v\n", 
		SumIntsOrFloats(ints), 
		SumIntsOrFloats(floats))
}
//SumIntsOrFloats 定义泛型方法
func SumIntsOrFloats[K comparable, V int64 | float64](m map[K]V) V {
	var s V
	for _, v := range m {
        s += v
    }
	return s
}

1.3 自定义泛型类型

  • any:代表 go里面所有的内置类型,等价于 interface {}
  • comparable:代表go里面内置的可比较类型:int、uint、float、bool、struct、指针等一切可比较类型
  • ~ 符号:用来表示改类型的衍生类型
//类型约束
type Number interface {
	int64 | float64
}
//进行类型约束时就可以使用当前类
func SumIntsNumbers[K comparable, V Number](m map[K]V) V {
	var s V
	for _, v := range m {
        s += v
	}
	return s
}

1.4 泛型与switch结合使用

func main() {
	fmt.Println(Get(12))
}
//go中不能直接将泛型与switch使用
func Get[T any](t T) T {
	var ti interface{} = &t
	switch v := ti.(type) {
	case *int:
		*v = 18
	}
	return t
}

1.5 泛型实战

使用泛型定义Json解析方法

//根据传入的类型通过反射获取到类型以及值
func typeFunc[E any](v any, e *E) *E {
	valueOf := reflect.ValueOf(v)
	typeOf := reflect.TypeOf(v)
	if k := typeOf.Kind(); k == reflect.Slice {
		json.Unmarshal(valueOf.Bytes(), e)
	}
	return e
}
func main() {
    user1 := &User{}
	user1 = typeFunc[User](marshal, user1)
	fmt.Printf("%+v", user1)
}

2. 反射

2.1 定义

Golang提供了一种机制,在编译时不知道类型的情况下,可更新变量、运行时查看值、调用方法以及直接对他们的布局进行操作的机制,称为反射。

2.2 方法

方法说明返回
reflect.ValueOf()获取输入参数接口中的数据的值,如果未空则返回 0,注意当前方法会使对象逃逸到堆空间当中返回的是 Value 对象
reflect.TypeOf()动态获取输入参数接口中的值的类型,如果为空则返回 nil返回的是 Type 对象

Value

type Value struct {
	typ *rtype
    //保存类型的值
	ptr unsafe.Pointer
    //指针类型
	flag
    //获取到值的指向地址,用于通过反射修改值
    Elem() Type
    //给value设置值
    Set()
}

Type

type Type interface {
    //根据索引获取到方法
	Method(int) Method
    //通过名称获取到方法
	MethodByName(string) (Method, bool)
    //获取到方法的数量
	NumMethod() int
    //获取结构名称
	Name() string
    //获取包路径
	PkgPath() string
    //获取到当前类型
	Kind() Kind
    //判断当前类型是否实现了接口
	Implements(u Type) bool
    //以位为单位返回类型的x
	Bits() int
    //获取到属性值的类型,类型必须是:Array、Chan、Map、Pointer、Slice,否则报错
	Elem() Type
    //获取到指定所以的值
	Field(i int) StructField
    //获取到对应索引的嵌套字段
	FieldByIndex(index []int) StructField
	//通过名称获取到对应的字段
	FieldByName(name string) (StructField, bool)
	FieldByNameFunc(match func(string) bool) (StructField, bool)
    .....
}

2.3 反射读取

func stringReflect() {
	name := "这是第一个反射字符串"
	valueOf := reflect.ValueOf(name)
	typeOf := reflect.TypeOf(name)
	fmt.Println(valueOf)
	fmt.Println(typeOf)
}

type Name struct {
	Name string
	Age string `use:"Ok"`
}
func (n Name) Show()  {
	fmt.Println(n.Name)
}
func structReflect() {
	name := Name{
		Name: "这是反射结构",
	}
	valueOf := reflect.ValueOf(name)
	typeOf := reflect.TypeOf(name)
	fmt.Printf("value值:%+v\n", valueOf)
	fmt.Printf("类型名称:%s\n", typeOf.Name())
	methodNum := typeOf.NumMethod()
	fmt.Printf("获取到方法的数量:%d", methodNum)
	for i := 0; i < methodNum; i++ {
		method := typeOf.Method(i)
		fmt.Printf("%v\t", method.Name)
	}
	fmt.Println()
	methodByName, _ := typeOf.MethodByName("Show")
	fmt.Printf("根据Show查找指定方法:%v\n", methodByName)
	//判断是否实现了当前接口,因为接口类型不能创建实例,所以把 nil 强制转为 *IName 类型
	implements := typeOf.Implements(reflect.TypeOf((*IName)(nil)).Elem())
	fmt.Printf("当前类型:%s,是否实现接口:%s,%v\n", typeOf.Name(), "IName", implements)
	fieldNum := typeOf.NumField()
	for i := 0; i < fieldNum; i++ {
		field := typeOf.Field(i)
		fmt.Printf("字段名称:%v\t", field.Name)
		if lookup, ok := field.Tag.Lookup("use") ; ok {
			fmt.Printf("获取标签:%v", lookup)
		}
	}
}

2.4 反射操作

func setValue() {
	name := Name{
		Name: "这是反射结构",
	}
	valueOf := reflect.ValueOf(&name)
	fmt.Printf("设置值之前:%+v\n", valueOf)
	//获取到地址值
	valueOf = valueOf.Elem()
	name1 := Name{
		Name: "这是通过反射设置的值",
	}
	valueOf.Set(reflect.ValueOf(name1))
	fmt.Printf("设置值之后:%+v\n", valueOf)
	//修改字段值
	fieldValueOf := valueOf.FieldByName("Name")
	fieldValueOf.SetString("这是修改字段之后的值")
	fmt.Printf("修改字段之后:%v\n", name.Name)
	//调用方法
	methodByName := valueOf.MethodByName("Show")
	values := make([]reflect.Value, 0)
	methodByName.Call(values)
}

2.5 判断

func judgeType() {
	name := Name{Name: "123"}
	typeOf := reflect.TypeOf(name)
	switch typeOf.Kind() {
	case reflect.Slice:
		fmt.Println("切面")
	case reflect.Array:
		fmt.Println("数组")
	case reflect.Struct:
		fmt.Println("结构体")
	}
    //判断具体类型
	var ti interface{} = &name
	switch ti.(type) {
	case *Name:
		fmt.Printf("%+v\n", ti)
	}
}

到此这篇关于Golang泛型与反射的应用详解的文章就介绍到这了,更多相关Golang泛型与反射内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
  • Golang 使用接口实现泛型的方法示例
  • Golang学习之反射机制的用法详解
  • Golang语言学习拿捏Go反射示例教程
  • Golang 利用反射对结构体优雅排序的操作方法
  • golang 如何用反射reflect操作结构体
  • golang 如何通过反射创建新对象
  • golang通过反射设置结构体变量的值
  • golang之反射和断言的具体使用
  • 详解Golang利用反射reflect动态调用方法

相关文章

  • Golang泛型与反射的应用详解

    Golang泛型与反射的应用详解

    目录1. 泛型1.1 定义1.2 例子1.3 自定义泛型类型1.4 泛型与switch结合使用1.5 泛型实战2. 反射2.1 定义2.2 方法2.3 反射读取2.4 反射操作2.5
    2022-06-15
  • C#使用Clipboard类实现剪贴板功能

    C#使用Clipboard类实现剪贴板功能

    剪贴板是Windows操作系统中最常用的功能之一,它用来从一个应用程序向另一个应用程序传递数据,可以是文本,图象,甚至是程序对象。 不过剪贴
    2022-06-15
  • 面试手写实现Promise.all

    面试手写实现Promise.all

    目录前言常见面试手写系列Promise.resolve简要回顾源码实现Promise.reject简要回顾源码实现Promise.all简要回顾源码实现Promise.allSettled简
    2022-06-15
  • Java中String和StringBuffer及StringBuilder?有什么区别

    Java中String和StringBuffer及StringBuilder?有什么区别

    目录String类为什么是immutable(不可变的)如何保证不可变string类为不可变对象的好处前言: String 是 Java 语言非常基础和重要的类,提供了
    2022-06-15
  • C#实现读写CSV文件的方法详解

    C#实现读写CSV文件的方法详解

    目录CSV文件标准文件示例RFC 4180简化标准读写CSV文件使用CsvHelper使用自定义方法总结项目中经常遇到CSV文件的读写需求,其中的难点主要是C
    2022-06-15
  • 一次性彻底讲透Python中pd.concat与pd.merge

    一次性彻底讲透Python中pd.concat与pd.merge

    目录数据拼接:pd.concat数据关联:pd.merge两者区别数据的合并与关联是数据处理过程中经常遇到的问题,在SQL、HQL中大家可能都有用到 join、
    2022-06-15
  • Python利用PyAutoGUI模块实现控制鼠标键盘

    Python利用PyAutoGUI模块实现控制鼠标键盘

    目录前言1、鼠标的相关控制2、键盘的相关控制前言 PyAutoGUI是一个简单易用,跨平台的可以模拟键盘鼠标进行自动操作的python库。 使用pip的方
    2022-06-15
  • 详解Android中motion_toast的使用

    详解Android中motion_toast的使用

    目录前言motion_toast 介绍示例最简单用法其他内置的提醒自定义 toast总结前言 我们通常会用 toast(也叫吐司)来显示提示信息,例如网络请求
    2022-06-15
  • Element如何实现loading的方法示例

    Element如何实现loading的方法示例

    目录前言使用 loading 的几种方式loading 指令实现指令通过指令来创建 loading代码实现directive创建 loading 实例loading 动画其他 loading
    2022-06-15
  • PTC Creo Schematics 9.0.0.0 中文授权激活版 Win64

    PTC Creo Schematics 9.0.0.0 中文授权激活版 Win64

    PTC Creo Schematics 9.0.0.0 中文授权激活版 Win64,PTC Creo Schematics 9.0中文破解版是一款领先的布线图软件,用于创建布线系统(如电缆、管道、HVAC 和液压系统)的 2D 示意图。该软件可根据 Creo Parametric 和 Creo Elements/Direct 中的现有 2D 原理图自动创建详细的 3D 布线系统设计
    2022-06-11

最新评论