反射操作普通变量

package main

import (
	"fmt"
	"reflect"
)

func main(){
	a := 1
	//reflect.TypeOf获取类型
	fmt.Println(reflect.TypeOf(a))   //int
	//查看占的字节
	fmt.Println(reflect.TypeOf(a).Size())//8
	//TypeOf获取的是reflect.Type类型,可以转化成字符串
	fmt.Println(reflect.TypeOf(a).Name() == "int") //true
	fmt.Println(reflect.TypeOf(a).String() == "int") //true

	//对于一个interface{},我们便可以通过这种方式来判断到底是什么类型
	func(val interface{}){
		switch t:=reflect.TypeOf(val).String();t {
		case "int":
			fmt.Println("val的类型为int")
		case "string":
			fmt.Println("val的类型为string")
		case "int32":
			fmt.Println("val的类型为int32")
		case "[]int":
			fallthrough
		case "[]string":
			fmt.Println("val的类型为切片")
		}
	}([]int{})  //val的类型为切片

	b := 'x'
	//获取的是reflect.Value类型,注意此时是没有办法比较的
	fmt.Println(reflect.ValueOf(b)) //120
	/*
	//获取相应的值,但你要提前知道类型,然后调用相应的方法
	fmt.Println(reflect.ValueOf(b).Int())
	fmt.Println(reflect.ValueOf(b).String())
	fmt.Println(reflect.ValueOf(b).Float())
	*/
	//不过也可以使用interface进行转化,会自动推断出具有真正类型的值
	fmt.Println(reflect.ValueOf(b).Interface() == 'x') //true
	//使用kind()或者Type()也可以获取值类型,再转换成String
	fmt.Println(reflect.ValueOf(b).Kind().String() == "int32")//true
	fmt.Println(reflect.ValueOf(b).Type().String() == "int32")//true

	//如何设置一个值
	name:="mashiro"
	//这里必须要传递指针进去,因为golang是值传递,传递值的话会单独拷贝一份。Elem()是为了得到变量的指针
	val := reflect.ValueOf(&name).Elem()
	fmt.Println(val.CanSet()) //true
	val.SetString("satori")
	fmt.Println(name)//satori

}

  

反射操作结构体

package main

import (
	"fmt"
	"reflect"
)

type Girl struct {
	Name string
	Age int
	Gender string
	Length int
}

//给结构体定义几个方法
func (g Girl)Info(){
	fmt.Println("info")
}
func (g Girl)Info1(){
	fmt.Println("info1")
}
func (g Girl)Info2(){
	fmt.Println("info2")
}

func main(){
	//先来如何获取结构体的类型
	g := Girl{"satori",16,"f",155}
	//我们之前看用Kind()和Type()获取的类型都是一样的,但那只是针对于int string之类的
	//当换成结构体的时候,我们看到,kind()的范围比较大,得到的是一个struct,而Type()得到的是具体的struct类型
	fmt.Println(reflect.ValueOf(g).Kind()) //struct
	fmt.Println(reflect.ValueOf(g).Type()) //main.girl

	//所以我们也可以判断是不是结构体,怎么判断呢?
	//注意使用Kind()得到的是一个reflect.Value类型,因此要是用reflect.Struct判断
	fmt.Println(reflect.ValueOf(g).Kind() == reflect.Struct) //true
	//如果使用字符串判断的话,那么可以转化一下
	fmt.Println(reflect.ValueOf(g).Kind().String() == "struct")//true

	//使用TypeOf(),Name()和String()获取的都是字符串,但是Name()不包含main
	fmt.Println(reflect.TypeOf(g).Name() == "girl")  //true
	fmt.Println(reflect.TypeOf(g).String() == "main.girl") //true

	//再来看看获取结构体内部元素的属性
	//如何获取结构体的字段的数量
	fmt.Println(reflect.ValueOf(g).NumField(), reflect.TypeOf(g).NumField())  //4 4
	//如何获取结构体的方法的数量,注意方法名必须要大写,否则检测不到
	fmt.Println(reflect.ValueOf(g).NumMethod(), reflect.TypeOf(g).NumMethod()) //3 3
	//调用方法,只能通过ValueOf,Call中必须传入[]reflect.Value类型
	var params []reflect.Value
	reflect.ValueOf(g).Method(0).Call(params)  //info
	reflect.ValueOf(g).Method(1).Call(params)  //info1
	reflect.ValueOf(g).Method(2).Call(params)  //info2

	//遍历结构体内部字段
	t:=reflect.TypeOf(g)
	v:=reflect.ValueOf(g)
	for i:=0;i<t.NumField();i++{
		//Field(i)获取第i个字段属性,Name则是获取名字。Type则是获取对应类型,如果比较的话,可以使用String()转成字符串类型
		fmt.Println("字段名:",t.Field(i).Name,"  值类型:", t.Field(i).Type.String())
		/*
		字段名: Name   值类型: string
		字段名: Age   值类型: int
		字段名: Gender   值类型: string
		字段名: Length   值类型: int
		 */
	}
	for i:=0;i<v.NumField();i++{
		fmt.Println("值类型:",v.Field(i).Type().String(), "  值类型:", v.Field(i).Kind().String(), " 值:", v.Field(i).Interface())
		/*
		值类型: string   值类型: string  值: satori
		值类型: int   值类型: int  值: 16
		值类型: string   值类型: string  值: f
		值类型: int   值类型: int  值: 155
		 */
	}
}

  

反射操作函数

package main

import (
	"fmt"
	"reflect"
	"runtime"
)

func foo(){}

func main(){
	t:=reflect.TypeOf(foo)
	v:=reflect.ValueOf(foo)
	fmt.Println(t.Name()) // 啥也没有
	fmt.Println(t.Kind(), t.Kind().String() == "func") // func  true
	fmt.Println(t.String(), t.String() == "func()") //func()  true

	fmt.Println(v.Type())//func()
	fmt.Println(v.Kind())// func

	//我们在结构体中,t.Name()打印了是girl,t.String()打印的main.girl
	//v.Type()打印的是main.girl。这里Kind()打印的是类型本身是一个func和结构体类似,但是v.Type和t.Name()和t.String()应该打印函数名啊
	//按照我们结构体的例子来说,t.Name()应该是foo,t.String()和v.Type()应该是main.foo才对啊,可这里不是,那我们如何获取函数名呢?
	fmt.Println(v.Interface())  //0x48e110, 这里打印的则是函数的内存地址,普通变量打印的是值

	//目前是拿不到函数名了,但是有没有其他的办法呢?虽然有点复杂,但还是可以拿到的
	pointer:=reflect.ValueOf(foo).Pointer()
	fmt.Println(runtime.FuncForPC(pointer).Name()) //main.foo
	fmt.Println(runtime.FuncForPC(pointer).Name() == "main.foo") //true

}

  

golang中的反射的更多相关文章

  1. golang中的反射reflect详解

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

  2. golang中的反射解析结构体标签tag

    package main import ( "fmt" "reflect" ) type resume struct { // 反射解析结构体标签tag Nam ...

  3. Golang通脉之反射

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

  4. 七、golang中接口、反射

    一.接口定义 1.定义 interface类型可以定义一组方法,但是这些不需要实现,并且interface不能包含任何变量 package main import ( "fmt" ...

  5. Golang中Struct与DB中表字段通过反射自动映射 - sqlmapper

    Golang中操作数据库已经有现成的库"database/sql"可以用,但是"database/sql"只提供了最基础的操作接口: 对数据库中一张表的增删改查 ...

  6. Go 中的反射要点

    简介 反射是元数据编程的一种形式,指的是程序获得本身结构的一种能力.不同语言的反射模型实现不一样,本文中的反射,仅仅指的是Go语言中的反射模型. 类型以及接口 这个基本概念需要清晰,这里不详细展开. ...

  7. Golang中的坑二

    Golang中的坑二 for ...range 最近两周用Golang做项目,编写web服务,两周时间写了大概五千行代码(业务代码加单元测试用例代码).用Go的感觉很爽,编码效率高,运行效率也不错,用 ...

  8. 【荐】详解 golang 中的 interface 和 nil

    golang 的 nil 在概念上和其它语言的 null.None.nil.NULL一样,都指代零值或空值.nil 是预先说明的标识符,也即通常意义上的关键字.在 golang 中,nil 只能赋值给 ...

  9. 六、golang中的结构体和方法、接口

    结构体: 1.用来自定义复杂数据结构 2.struct里面可以包含多个字段(属性) 3.struct类型可以定义方法,注意和函数的区分 4.strucr类型是值类型 5.struct类型可以嵌套 6. ...

随机推荐

  1. debian8.2安装kubernetes

    master上通过kubeadm安装Kubernetes 添加国内阿里源后安装kubeadm: deb https://mirrors.aliyun.com/kubernetes/apt/ kuber ...

  2. TLS协议分析

    TLS协议分析 本文目标: 学习鉴赏TLS协议的设计,透彻理解原理和重点细节 跟进一下密码学应用领域的历史和进展 整理现代加密通信协议设计的一般思路 本文有门槛,读者需要对现代密码学有清晰而系统的理解 ...

  3. Windows IRP

    IRP(I/O Request Packet),是由IO manager发起的对device的IO请求. 当用户调用系统API,如createFile类似的函数,其实是会交给IO manager来做相 ...

  4. Windows下的Memcache安装与Java部署

    Windows下的Memcache安装: 1. 下载memcached的windows稳定版,解压放某个盘下面,比如在c:\memcached 2. 在终端(也即cmd命令界面)下输入 ‘c:\mem ...

  5. hihocoder 1323 回文字符串(字符串+dp)

    题解: 比较水的题目 dp[i][j]表示[i...j]最少改变几次变成回文字符串 那么有三种转移 dp[i][j] = dp[i+1][j-1] + s[i] != s[j] dp[i][j] = ...

  6. 网络编程:listen函数

    listen函数仅由TCP服务器调用,它做两件事: 当socket函数创建一个套接字时,它被假设为一个主动套接字,也就是说,它是一个将调用connect发起连接的客户套接字.listen函数把一个未连 ...

  7. POJ2236:Wireless Network(并查集)

    Wireless Network Time Limit: 10000MS   Memory Limit: 65536K Total Submissions: 39772   Accepted: 164 ...

  8. Create a conditional DNS forwarder on our domain.com to Amazon default DNS provider

    Backgroup: I have an AWS Managed Active Directory(domain.com). I created a DHCP options set  to my d ...

  9. GTK+与MFC不完全对比

    转载自:http://tech.ddvip.com/2007-11/119640973738229.html 1. 两者都是基于面向对象设计的.尽管MFC是用C++写的,而GTK+是用C写的,但思想都 ...

  10. HDU1025---(LIS 最长上升子序列 的应用)

    分析: n行 每行包含两个整数p r;意思是p从到r 不能有交叉的路 p刚好从1->n, 可看做下标,到的地方看做值 就转化为了最长上升子序列的问题 此题难点,怎么将其转化为LIS问题 #inc ...