golang 反射,泛型参数类型约束,泛型结构体约束,单元测试综合示例

golang中有关反射, 泛型约束和单元测试的综合案例,废话不多说直接上代码

文件名:modifydemo/modify.go

Go 复制代码
package modifydemo

import (
	"fmt"
	"reflect"
)

type User struct {
	UserId int    `json:"uid"`
	Name   string `json:"name"`
}
type Student struct {
	Name   string `json:"name"`
	UserId int    `json:"uid"`
}

// 定义一个自定义的类型接口 用来约束要修改的新值的范围,只能是这里支持的值
type MyType interface {
	int | int8 | int32 | int64 | uint | uint32 | uint64 | float32 | float64 | ~string | bool
}

// 反射修改结构体演示函数,
// 这里的第1个和第3个参数都使用泛型进行了约束,
// 即 参数obj只能是 User或者Student且必须是指针
// 参数 newVal 只能是上面的MyType接口中定义的类型之一
func ModifyStructDemo[T MyType, M ~*S, S User | Student](obj M, name string, newVal T) (err error) {
	//捕获panic异常 单元测试时不要捕获, 如果是生产环境的话就需要捕获这里可能的panic异常,避免影响后续程序的运行
	// defer func() {
	// 	if err := recover(); err != nil {
	// 		fmt.Println(err)
	// 	}
	// }()

	// 这个地方需要注意区分各个rv (reflect.Value), 因为ValueOf(), Elem(), FieldByName()他们返的对象都是 rv,
	// 但是他们各自对应的实际对象又都不同, 一定要分清楚,特别是名称不能乱用,否则就 会造成混乱!
	rv := reflect.ValueOf(obj)                       // 要修改对象的 rv
	rvElemFieldByName := rv.Elem().FieldByName(name) // 这时要修改对象上的字段的 rv

	// 如果字段name对应的rv不合法,这返回异常
	if !rvElemFieldByName.IsValid() {
		return fmt.Errorf("字段 %v 在 %T 中不存在", name, obj)
	}

	rvNewVal := reflect.ValueOf(newVal) // 这个地方是只需要获取新值对应的rv就行,不要和前面的对象关联,否则原来的rv就会被改变

	// 获取要修改的字段的reflect.Value.Kind() 类别用于switch中对不同的值调用rvf对应的方法进行修改
	newValKind := rvNewVal.Kind()
	// 根据要修改的值的类型进行匹配
	switch newValKind {
	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
		nv := reflect.ValueOf(newVal).Int() // 这里通过反射将所有上面的int类型全部转换为int64类型
		rvElemFieldByName.SetInt(nv)        // 修改为新值
	case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
		nv := reflect.ValueOf(newVal).Uint() // 这里通过反射将所有上面的uint类型全部转换为uint64类型
		rvElemFieldByName.SetUint(nv)        // 修改为新值
	case reflect.Float32, reflect.Float64:
		nv := reflect.ValueOf(newVal).Float()
		rvElemFieldByName.SetFloat(nv)
	case reflect.String:
		nv := reflect.ValueOf(newVal).String()
		rvElemFieldByName.SetString(nv)
	case reflect.Bool:
		nv := reflect.ValueOf(newVal).Bool()
		rvElemFieldByName.SetBool(nv)
	default:
		err = fmt.Errorf("要设置的值类型 %T 不被支持", newVal)
	}
	return
}

单元测试 测试用例文件名:modifydemo/modify_test.go

Go 复制代码
package modifydemo

import (
	"testing"
)

func TestModifyStructDemo(t *testing.T) {
	user := &User{Name: "alex", UserId: 1}
	t.Logf("修改前的用户信息: %v", user)
	err := ModifyStructDemo(user, "Name", "p")
	if err != nil {
		t.Fatal(err)
	}
	t.Logf("修改后的用户信息: %v", user)

	stu := &Student{Name: "小明", UserId: 18}
	t.Logf("修改前的stu信息: %v", stu)
	err2 := ModifyStructDemo(stu, "Name", "小张")
	if err2 != nil {
		t.Fatal(err2)
	}
	t.Logf("修改后的stu信息: %v", stu)
}

运行单元测试 go test -v

相关推荐
阿珊和她的猫4 小时前
v-scale-scree: 根据屏幕尺寸缩放内容
开发语言·前端·javascript
fouryears_234176 小时前
Flutter InheritedWidget 详解:从生命周期到数据流动的完整解析
开发语言·flutter·客户端·dart
我好喜欢你~7 小时前
C#---StopWatch类
开发语言·c#
lifallen8 小时前
Java Stream sort算子实现:SortedOps
java·开发语言
IT毕设实战小研8 小时前
基于Spring Boot 4s店车辆管理系统 租车管理系统 停车位管理系统 智慧车辆管理系统
java·开发语言·spring boot·后端·spring·毕业设计·课程设计
apocelipes9 小时前
下划线字段在golang结构体中的应用
golang
cui__OaO10 小时前
Linux软件编程--线程
linux·开发语言·线程·互斥锁·死锁·信号量·嵌入式学习
鱼鱼说测试10 小时前
Jenkins+Python自动化持续集成详细教程
开发语言·servlet·php
艾莉丝努力练剑10 小时前
【洛谷刷题】用C语言和C++做一些入门题,练习洛谷IDE模式:分支机构(一)
c语言·开发语言·数据结构·c++·学习·算法
CHEN5_0211 小时前
【Java基础面试题】Java基础概念
java·开发语言