golang 通过reflect反射修改值
不是所有的反射值都可以修改。对于一个反射值是否可以修改,可以通过CanSet()
进行检查。
要修改值,必须满足:
可以寻址
可寻址的类型:
- 指针指向的具体元素
- slice的元素
- 可寻址的结构体的字段(指向结构体的指针)
- 可寻址的数组的元素(指向数组的指针)
不是结构体没有导出的字段
1.指针指向的具体元素
需要两步:
- 取地址:
v := reflect.ValueOf(&x)
- 取得具体值
v=v.Elem()
下面通过一个整型变量的赋值进行说明。
package main
import (
"fmt"
"reflect"
)
func main() {
a := 1
v := reflect.ValueOf(a)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
v = reflect.ValueOf(&a)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
v = v.Elem() // element value
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
// set
v.SetInt(2)
fmt.Println("after set, v:", v)
newValue := reflect.ValueOf(3)
v.Set(newValue)
fmt.Println("after set, v:", v)
}
定义一个整型变量,得到反射值,检查是否可以修改。
接着,通过反射获取整型变量地址的反射值,再通过Elem()
得到指针指向的具体对象,这样就可以修改值。
输出结果:
v: 1
v Type: int
v CanSet: false
v Type: *int
v CanSet: false
v Type: int
v CanSet: true
after set, v: 2
after set, v: 3
2.slice的元素
需要两步:
- 取切片:
v := reflect.ValueOf(s)
- 获取切片的元素
e := v.Index(0)
具体代码如下。
package main
import (
"fmt"
"reflect"
)
func main() {
a := []int{1,1}
v := reflect.ValueOf(a)
fmt.Println("v:", v)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
e := v.Index(0)
fmt.Println("e CanSet:", e.CanSet())
e.SetInt(2)
fmt.Println("after set:", v)
}
定义一个slice,得到反射值,从slice 的反射值中元素,
接着,修改元素的值。
结果输出:
v: [1 1]
v Type: []int
v CanSet: false
e CanSet: true
after set: [2 1]
3.可寻址的结构体的字段
需要三步:
- 取结构体地址
v := reflect.ValueOf(&a)
- 获取结构体的具体值:
Elem()
- 获取结构体的字段:
FieldByName()
package main
import (
"fmt"
"reflect"
)
type Orange struct {
Size int
}
func main() {
a := Orange{99}
v := reflect.ValueOf(a)
fmt.Println("v:", v)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
v = reflect.ValueOf(&a)
fmt.Println("v:", v)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
//element
v = v.Elem()
size := v.FieldByName("Size")
fmt.Println("size CanSet:", size.CanSet())
size.SetInt(88)
fmt.Println("after set:", v)
}
定义一个结构体变量,首先反射变量,检查是否可以修改值。
接着,使用变量地址进行反射,通过Elem()
获取指针指向的具体值。
最后,获取结构体的字段,并修改值。
输出结果
v: {99}
v Type: main.Orange
v CanSet: false
v: &{99}
v Type: *main.Orange
v CanSet: false
size CanSet: true
after set: {88}
4.可寻址的数组的元素
需要三步:
- 取数组地址
v := reflect.ValueOf(&a)
- 获取反射对象中的具体值:
Elem()
- 通过索引获取元素:
Index(0)
package main
import (
"fmt"
"reflect"
)
func main() {
a := [2]int{1,1}
v := reflect.ValueOf(a)
fmt.Println("v:", v)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
e := v.Index(0)
fmt.Println("e Type:", e.Type())
fmt.Println("e CanSet:", e.CanSet())
v = reflect.ValueOf(&a)
fmt.Println("v:", v)
fmt.Println("v Type:", v.Type())
fmt.Println("v CanSet:", v.CanSet())
// element
v = v.Elem()
e = v.Index(0)
fmt.Println("e Type:", e.Type())
fmt.Println("e CanSet:", e.CanSet())
e.SetInt(3)
fmt.Println("after set:", v)
}
定义一个数组变量,首先反射变量,检查是否可以修改值。
接着,取数组中的元素,检查是否可以修改值。
使用数组的地址进行反射,通过Elem()
获取指针指向的具体值。
最后,获取数组中的元素,并修改值。
输出结果
v: [1 1]
v Type: [2]int
v CanSet: false
e Type: int
e CanSet: false
v: &[1 1]
v Type: *[2]int
v CanSet: false
e Type: int
e CanSet: true
after set: [3 1]
golang 通过reflect反射修改值的更多相关文章
- golang:reflect反射
因为之前一直以C++为主要开发语言,所以刚接触go语言中的reflect时感觉很懵逼,因此决定找资料彻底学习一下. 到底反射是什么? https://blog.golang.org/laws-of-r ...
- golang基础--reflect反射
反射的知识点比较晦涩,后期会对此知识点展开深入的分析及示例代码展示 反射可达大提高程序的灵活性,使得inferface{}有更大的发挥余地 反射使用TypeOf和ValueOf函数从接口中获取目标对象 ...
- golang 使用reflect反射结构体
"反射结构体"是指在程序执行时,遍历结构体中的字段以及方法. 1.反射结构体 下面使用一个简单的例子说明如何反射结构体. 定义一个结构体,包括3个字段,以及一个方法. 通过refl ...
- go中的关键字-reflect 反射
1. 什么是反射 Golang提供了一种机制,在编译时不知道类型的情况下,可更新变量.运行时查看值.调用方法以及直接对他们的布局进行操作的机制,称为反射. 2. 反射的使用 2.1 获取变量内部信息 ...
- golang中的反射reflect详解
先重复一遍反射三定律: 1.反射可以将"接口类型变量"转换为"反射类型对象". 2.反射可以将"反射类型对象"转换为"接口类型变量 ...
- Golang 接口与反射知识要点
目录 Golang 接口与反射知识要点 1. 接口类型变量 2. 类型断言 3. 鸭子类型 4. 反射机制 5. reflect 包 TypeOf().ValueOf() Type().Kind() ...
- Golang通脉之反射
什么是反射 官方关于反射定义: Reflection in computing is the ability of a program to examine its own structure, pa ...
- JAVA 构造器, extends[继承], implements[实现], Interface[接口], reflect[反射], clone[克隆], final, static, abstrac
记录一下: 构造器[构造函数]: 在java中如果用户编写类的时候没有提供构造函数,那么编译器会自动提供一个默认构造函数.它会把所有的实例字段设置为默认值:所有的数字变量初始化为0;所有的布尔变量设置 ...
- mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样
Mybatis批量更新数据 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批 ...
随机推荐
- SQL-Serverの自動採番(IDENTITY値)の取得・リセット
システムに必要なテーブルで.自動的に番号を振っていくものが必要なときがあります. たとえば.各種の伝票データの伝票番号の様なものです. プログラム処理上.データを登録した直後に.自動採番された値を取得 ...
- Java高并发程序设计学习笔记(二):多线程基础
转自:https://blog.csdn.net/dataiyangu/article/details/86226835# 什么是线程?线程的基本操作线程的基本操作新建线程调用run的一种方式调用ru ...
- 自学Python5.7-面向对象三大基本特征_封装
自学Python之路-Python基础+模块+面向对象自学Python之路-Python网络编程自学Python之路-Python并发编程+数据库+前端自学Python之路-django 自学Pyth ...
- JMeter元件之Test Fragment
简介 JMeter中的Test Fragment,是控制器上一个种特殊的线程组,它在测试树上与线程组处于同一层级.但使用时需要和include Controller或者Module Controlle ...
- C++ 编码导致编译错误
记录一个问题 平时经常用Notepad++写C++,然后g++编译 最近出现一个莫名其妙的问题 原来是编译时遇到了非法字符 用Notepad++一看,原来文件的编码出现的变化 不是UTF-8或者其他对 ...
- centos7安装es
#安装java1.8rpm -ivh jdk-8u191-linux-x64.rpm #解压estar -zxvf elasticsearch-6.4.0.tar.gz -C /usr #修改es限制 ...
- 如何使用git回退部分修改(转)
如何使用git回退部分修改(转) 很多时候,git新手容易误操作,比如,在levelIISZ-1.4.dev分支下,运行了git pull idc cpp-1.0的结果,这样做麻烦很大,经常导 ...
- Acwing-278-数字组合(背包)
链接: https://www.acwing.com/problem/content/280/ 题意: 给定N个正整数A1,A2,-,AN,从中选出若干个数,使它们的和为M,求有多少种选择方案. 思路 ...
- Java基础——集合框架(待整理)
ArrayList 和 和 Vector 的区别 从代码的最终的操作形式上可以发现,代码的输出结果与之前是一样的,而且没有区别,但是两者的区别还在于其内部的组成上. No. 区别点 Vector Ve ...
- 【转】推荐几本学习MySQL的好书-MySQL 深入的书籍
MySQL的使用 1 MySQL技术内幕InnoDB存储引擎 2 MySQL的官方手册 3 MySQL排错指南 4 高性能MySQL 5 数据库索引设计与优化 6 Effective MySQL系列 ...