Golang泛型与反射的应用详解
来源:脚本之家
时间:2022-12-24 09:21:53 112浏览 收藏
怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《Golang泛型与反射的应用详解》,涉及到反射、泛型,有需要的可以收藏一下
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
<h3>2.4 反射操作</h3>
<pre class="brush:go;">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学习网公众号!
声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
174 收藏
-
246 收藏
-
353 收藏
-
234 收藏
-
346 收藏
最新阅读
更多>
-
110 收藏
-
412 收藏
-
423 收藏
-
274 收藏
-
379 收藏
-
241 收藏
-
235 收藏
-
365 收藏
-
247 收藏
-
241 收藏
-
467 收藏
-
500 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习