golang学习笔记27-反射【重要】

本节也是GO核心部分,很重要。包括基本类型的反射,结构体类型的反射,类别方法Kind(),修改变量的值。

一、概念,基本类型的反射

【1】反射可以做什么?
1)反射可以在运行时动态获取变量的各种信息,比如变量的类型,类别等信息
2)如果是结构体变量,还可以获取到结构体本身的信息(包括结构体的字段、方法)
3)通过反射,可以修改变量的值,可以调用关联的方法。
4)使用反射,需要import "reflect"
【2】反射相关的函数
1)reflect.TypeOf(变量名),获取变量的类型,返回reflect.Type类型
2)reflect.ValueOf(变量名),获取变量的值,返回reflect.Value类型
反射不仅可以获取变量名和变量类型,reflect.Type也可以通过空接口转回原类型:

package main

import (
	"fmt"
	"reflect"
)

func main() {
	// 定义一个变量
	var x int = 42

	// 获取变量的类型
	t := reflect.TypeOf(x)
	fmt.Println("Type:", t) // 输出: Type: int

	// 获取变量的值
	v := reflect.ValueOf(x)
	fmt.Println("Value:", v) // 输出: Value: 42

	// 将 reflect.Value 转换回原始类型
	// Step 1: 将 reflect.Value 转换为 empty interface (interface{})
	emptyInterface := v.Interface() // 这里使用空接口可以接受任何类型的值
	// Step 2: 使用类型断言将 empty interface 转换回原始类型 int
	originalValue := emptyInterface.(int)         // 将空接口断言为 int 类型
	fmt.Println("Original value:", originalValue) // 输出: Original value: 42
}

反射和数据类型互转的流程图如下:
在这里插入图片描述

二、结构体类型的反射

和基本类型的情况差不多,但要注意因为实现接口的结构体可能有多个,接口转结构体要判断是否转成功:

package main

import (
	"fmt"
	"reflect"
)

// 定义 student 结构体
type student struct {
	Name string
	Age  int
}

func main() {
	// 创建一个 student 实例
	s := student{Name: "Alice", Age: 20}

	// 获取变量的类型
	t := reflect.TypeOf(s)
	fmt.Println("类型:", t) // 输出: 类型: main.student

	// 获取变量的值
	v := reflect.ValueOf(s)
	fmt.Println("值:", v) // 输出: 值: {Alice 20}

	// 将 reflect.Value 转换回原始类型
	// Step 1: 将 reflect.Value 转换为 empty interface (interface{})
	emptyInterface := v.Interface() // 这里使用空接口可以接受任何类型的值

	// Step 2: 使用类型断言将 empty interface 转换回原始类型 student
	originalStudent, ok := emptyInterface.(student) // 将空接口断言为 student 类型
	if ok {
		// 如果转换成功,输出姓名和年龄
		fmt.Printf("原始学生 - 姓名: %s, 年龄: %d\n", originalStudent.Name, originalStudent.Age) // 输出: 原始学生 - 姓名: Alice, 年龄: 20
	} else {
		fmt.Println("类型断言为 student 失败。")
	}
}

三、类别方法Kind()

Kind()是reflect.Type的一个方法,用于获取类型的基本种类(kind)。它返回一个reflect.Kind类型的值,用于描述基本数据类型的特性,如int、string、struct等。
Kind()和TypeOf()的区别如下表所示:

特性reflect.TypeOf()reflect.Kind()
返回值返回 reflect.Type 类型的对象返回 reflect.Kind 类型的枚举值
作用获取变量的完整类型信息获取变量的基本种类(如 intstringstruct
适用场景当需要获取类型的详细信息时当只需要判断数据类型的基本特性时

语法:TypeOf(s).Kind()ValueOf(s).Kind(),这两个操作都返回变量s的基本类型。

四、修改变量的值

如果用反射修改x的类型,需要先获取reflect.Value类型,然后用对应x类型的方法,比如SetInt(),如果x是int*,则需要先用Elem(),再用SetInt():

package main

import (
	"fmt"
	"reflect"
)

func main() {
	var x int = 42
	p := &x // 创建指向 x 的指针

	// 获取指针的 reflect.Value
	v := reflect.ValueOf(p)

	// 使用 Elem() 获取指针指向的值
	elem := v.Elem()

	// 修改指针指向的值
	elem.SetInt(100)

	// 输出修改后的值
	fmt.Println("修改后的值:", x) // 输出: 修改后的值: 100
}

如果x是结构体,要用Field()获取字段,Method()获取方法,用reflect.Value切片调用有参方法,用nil调用无参方法:

package main

import (
	"fmt"
	"reflect"
)

// 定义 student 结构体
type student struct {
	Name string
	Age  int
}

// 为 student 结构体定义一个方法
func (s *student) SetAge(age int) {
	s.Age = age
}

// 为 student 结构体定义另一个方法
func (s *student) GetInfo() string {
	return fmt.Sprintf("姓名: %s, 年龄: %d", s.Name, s.Age)
}

func main() {
	// 创建一个 student 实例
	s := student{Name: "Alice", Age: 20}

	// 获取结构体的类型,使用指针获取
	stuType := reflect.TypeOf(&s)

	// 获取字段数量
	numFields := stuType.Elem().NumField() // 使用 Elem() 获取底层类型
	fmt.Printf("字段数量: %d\n", numFields)

	// 遍历字段
	for i := 0; i < numFields; i++ {
		field := stuType.Elem().Field(i) // 使用 Elem() 获取底层类型的字段
		fmt.Printf("字段名: %s, 字段类型: %s\n", field.Name, field.Type)
	}

	// 获取方法数量
	numMethods := stuType.NumMethod() // 获取方法数量
	fmt.Printf("方法数量: %d\n", numMethods)

	// 遍历方法
	for i := 0; i < numMethods; i++ {
		method := stuType.Method(i)
		fmt.Printf("方法名: %s\n", method.Name)
	}

	// 使用反射修改 Name 字段的值
	stuValue := reflect.ValueOf(&s)       // 获取结构体的反射值,使用指针可以修改值
	nameField := stuValue.Elem().Field(0) // 获取第一个字段的反射值

	// 确保字段可设置
	if nameField.CanSet() {
		nameField.SetString("Bob") // 修改 Name 字段的值为 "Bob"
	}

	// 调用 SetAge 方法,将年龄设置为 30
	setAgeMethod := stuValue.MethodByName("SetAge")
	args := []reflect.Value{reflect.ValueOf(30)} // 创建包含参数的切片
	setAgeMethod.Call(args)                      // 调用 SetAge 方法,传入参数

	// 调用 GetInfo 方法
	getInfoMethod := stuValue.MethodByName("GetInfo")
	info := getInfoMethod.Call(nil) // 调用方法,传递空参数

	// 输出信息
	fmt.Println(info[0]) // 输出: 姓名: Bob, 年龄: 30
}

关键代码解释:
1.info := getInfoMethod.Call(nil)
infoMethod是通过反射获取到的一个方法的反射值。在这个例子中,它指向student结构体的Info方法。Call是reflect.Value类型的方法,用于调用一个方法。它接受一个参数,参数是一个reflect.Value切片,表示要传递给被调用方法的参数。在这里,我们传递了nil,表示Info方法不需要任何参数。在这个例子中,GetInfo方法返回一个字符串,因此info将是一个包含一个reflect.Value的切片,表示学生信息字符串。
2. args := []reflect.Value{reflect.ValueOf(30)}
这一行创建了一个reflect.Value切片,命名为args,它将用于调用SetAge方法。reflect.ValueOf(30)用于将整数30转换为reflect.Value类型。[]reflect.Value{}表示创建一个reflect.Value类型的切片,作为SetAge方法的参数。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术卷

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值