不是所有的反射值都可以修改。对于一个反射值是否可以修改,可以通过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反射修改值的更多相关文章

  1. golang:reflect反射

    因为之前一直以C++为主要开发语言,所以刚接触go语言中的reflect时感觉很懵逼,因此决定找资料彻底学习一下. 到底反射是什么? https://blog.golang.org/laws-of-r ...

  2. golang基础--reflect反射

    反射的知识点比较晦涩,后期会对此知识点展开深入的分析及示例代码展示 反射可达大提高程序的灵活性,使得inferface{}有更大的发挥余地 反射使用TypeOf和ValueOf函数从接口中获取目标对象 ...

  3. golang 使用reflect反射结构体

    "反射结构体"是指在程序执行时,遍历结构体中的字段以及方法. 1.反射结构体 下面使用一个简单的例子说明如何反射结构体. 定义一个结构体,包括3个字段,以及一个方法. 通过refl ...

  4. go中的关键字-reflect 反射

    1. 什么是反射 Golang提供了一种机制,在编译时不知道类型的情况下,可更新变量.运行时查看值.调用方法以及直接对他们的布局进行操作的机制,称为反射. 2. 反射的使用 2.1 获取变量内部信息 ...

  5. golang中的反射reflect详解

    先重复一遍反射三定律: 1.反射可以将"接口类型变量"转换为"反射类型对象". 2.反射可以将"反射类型对象"转换为"接口类型变量 ...

  6. Golang 接口与反射知识要点

    目录 Golang 接口与反射知识要点 1. 接口类型变量 2. 类型断言 3. 鸭子类型 4. 反射机制 5. reflect 包 TypeOf().ValueOf() Type().Kind() ...

  7. Golang通脉之反射

    什么是反射 官方关于反射定义: Reflection in computing is the ability of a program to examine its own structure, pa ...

  8. JAVA 构造器, extends[继承], implements[实现], Interface[接口], reflect[反射], clone[克隆], final, static, abstrac

    记录一下: 构造器[构造函数]: 在java中如果用户编写类的时候没有提供构造函数,那么编译器会自动提供一个默认构造函数.它会把所有的实例字段设置为默认值:所有的数字变量初始化为0;所有的布尔变量设置 ...

  9. mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样

    Mybatis批量更新数据 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批 ...

随机推荐

  1. Idea java 程序打jar包(maven)

    1.准备好控制台程序 2.引用的项目打包(公共类接口) 3.开发打包 点击运行 打包结果如下

  2. 网络初级篇之STP(概念原理)

    一.什么是STP 生成树协议(Spanning Tree Protocol,STP),是一种工作在OSI网络模型中的第二层(数据链路层)的通信协议,基本应用是防止交换机冗余链路产生的环路.用于确保以太 ...

  3. Java 实现《编译原理》简单-语法分析功能-LL(1)文法 - 程序解析

    Java 实现<编译原理>简单-语法分析功能-LL(1)文法 - 程序解析 编译原理学习,语法分析程序设计 (一)要求及功能 已知 LL(1) 文法为: G'[E]: E→TE' E'→+ ...

  4. iOS中为控件设置颜色渐变和透明度渐变

    项目中用到地图设置渐变色,查找资料找到两种方法:一种设置颜色,一种设置透明度: //为颜色设置渐变效果: UIView *view = [[UIView alloc] initWithFrame:CG ...

  5. 优秀.NET界面控件DevExpress v19.1.6全新来袭!新改进抢“鲜”看

    DevExpress Universal Subscription(又名DevExpress宇宙版或DXperience Universal Suite)是全球使用广泛的.NET用户界面控件套包,De ...

  6. ES6学习笔记(对象新增方法)

    1.Object.is() ES5 比较两个值是否相等,只有两个运算符:相等运算符(==)和严格相等运算符(===).它们都有缺点,前者会自动转换数据类型,后者的NaN不等于自身,以及+0等于-0. ...

  7. 扩展Puppet – 建立Puppet CA集群

    扩展Puppet – 建立Puppet CA集群  (1 votes, average: 5.00 out of 5) 588 views 2012 年 3 月 4 日Puppet.运维ca.mast ...

  8. fedora29 安装mongodb 4.0,6问题记录

    如果运行mongod命令时提示 无加载共享库libcrypto.so.10,那就到页面下载http://www.rpmfind.net/linux/rpm2html/search.php?query= ...

  9. Python:n个点的费马问题

    问题描述 在平面内有n(n>=3)个点N1(x1,y1),N2(x2,y2),...,Nn(xn,yn),现求一点P(x,y),使得P到各点直线距离之和最小. 算法分析 当n=3时,这是著名的三 ...

  10. 基于Python原生asyncio模块对DNS正向和反向的解析

    一.正向解析:域名解析IP地址 import asyncio import socket domains = [ ('www.baidu.com', 'https'), ('cn.bing.com', ...