• Title


    通过反射获取值

    func reflectValue(x interface{}) {
    	v := reflect.ValueOf(x)
    	k := v.Kind()
    	switch k {
    	case reflect.Int64:
    		// v.Int()从反射中获取整型的原始值,然后通过int64()强制类型转换
    		fmt.Printf("type is int64, value is %d
    ", int64(v.Int()))
    	case reflect.Float32:
    		// v.Float()从反射中获取浮点型的原始值,然后通过float32()强制类型转换
    		fmt.Printf("type is float32, value is %f
    ", float32(v.Float()))
    	case reflect.Float64:
    		// v.Float()从反射中获取浮点型的原始值,然后通过float64()强制类型转换
    		fmt.Printf("type is float64, value is %f
    ", float64(v.Float()))
    	}
    }
    func main() {
    	var a float32 = 3.14
    	var b int64 = 100
    	reflectValue(a) // type is float32, value is 3.140000
    	reflectValue(b) // type is int64, value is 100
    	// 将int类型的原始值转换为reflect.Value类型
    	c := reflect.ValueOf(10)
    	fmt.Printf("type c :%T
    ", c) // type c :reflect.Value
    }
    

    通过反射修改变量的值

    函数传递的参数是值拷贝
    go反射中使用 Elem()方法获取指针对应的值

    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func reflectSetValue1(x interface{}) {
    	v := reflect.ValueOf(x)
    	if v.Kind() == reflect.Int64 {
    		v.SetInt(200) //修改的是副本,reflect包会引发panic
    	}
    }
    func reflectSetValue2(x interface{}) {
    	v := reflect.ValueOf(x)
    	// 反射中使用 Elem()方法获取指针对应的值
    	if v.Elem().Kind() == reflect.Int64 {
    		v.Elem().SetInt(200)
    	}
    }
    func main() {
    	var a int64 = 100
    	// reflectSetValue1(a) //panic: reflect: reflect.Value.SetInt using unaddressable value
    	reflectSetValue2(&a)
    	fmt.Println(a)
    }
    

    IsNil()和IsValid()

    IsNil()常被用于判断指针是否为空;IsValid()常被用于判定返回值是否有效

    func main() {
    	// *int类型空指针
    	var a *int
    	fmt.Println("var a *int IsNil:", reflect.ValueOf(a).IsNil())
    	// nil值
    	fmt.Println("nil IsValid:", reflect.ValueOf(nil).IsValid())
    	// 实例化一个匿名结构体
    	b := struct{}{}
    	// 尝试从结构体中查找"abc"字段
    	fmt.Println("不存在的结构体成员:", reflect.ValueOf(b).FieldByName("abc").IsValid())
    	// 尝试从结构体中查找"abc"方法
    	fmt.Println("不存在的结构体方法:", reflect.ValueOf(b).MethodByName("abc").IsValid())
    	// map
    	c := map[string]int{}
    	// 尝试从map中查找一个不存在的键
    	fmt.Println("map中不存在的键:", reflect.ValueOf(c).MapIndex(reflect.ValueOf("娜扎")).IsValid())
    }
    

    go的结构体反射

    任意值通过reflect.TypeOf()获得反射对象信息后,如果它的类型是结构体,可以通过反射值对象的NumField()和Field()方法获得结构体成员的详细信息

    type student struct {
    	Name  string `json:"name"`
    	Score int    `json:"score"`
    }
    
    func main() {
    	stu1 := student{
    		Name:  "alex",
    		Score: 90,
    	}
    
    	t := reflect.TypeOf(stu1)
    	fmt.Println(t.Name(), t.Kind()) // student struct
    	// 通过for循环遍历结构体的所有字段信息
    	for i := 0; i < t.NumField(); i++ {
    		field := t.Field(i)
    		fmt.Printf("name:%s index:%d type:%v json tag:%v
    ", field.Name, field.Index, field.Type, field.Tag.Get("json"))
    	}
    
    	// 通过字段名获取指定结构体字段信息
    	if scoreField, ok := t.FieldByName("Score"); ok {
    		fmt.Printf("name:%s index:%d type:%v json tag:%v
    ", scoreField.Name, scoreField.Index, scoreField.Type, scoreField.Tag.Get("json"))
    	}
    }
    

    go通过反射获取结构体的方法

    // 给student添加两个方法 Study和Sleep(注意首字母大写)
    func (s student) Study() string {
    	msg := "好好学习,天天向上。"
    	fmt.Println(msg)
    	return msg
    }
    
    func (s student) Sleep() string {
    	msg := "好好睡觉,快快长大。"
    	fmt.Println(msg)
    	return msg
    }
    
    func printMethod(x interface{}) {
    	t := reflect.TypeOf(x)
    	v := reflect.ValueOf(x)
    
    	fmt.Println(t.NumMethod())
    	for i := 0; i < v.NumMethod(); i++ {
    		methodType := v.Method(i).Type()
    		fmt.Printf("method name:%s
    ", t.Method(i).Name)
    		fmt.Printf("method:%s
    ", methodType)
    		// 通过反射调用方法传递的参数必须是 []reflect.Value 类型
    		var args = []reflect.Value{}
    		v.Method(i).Call(args)
    	}
    }
    
  • 相关阅读:
    minio 对于压缩的处理
    mino federation 功能
    Full Schema Stitching with Apollo Server
    GraphQL Gateway Architectures
    Modularizing your graphQL schemas
    gearman openresty 集成试用
    madlib 集成 hasura graphql-engine 试用
    Oracle数据库--解决单张表中数据量巨大(大数据、数据量上百万级别,后查询,更新数据等耗时剧增)
    绝对干货,教你4分钟插入1000万条数据到mysql数据库表,快快进来
    几款开源的ETL工具及ELT初探
  • 原文地址:https://www.cnblogs.com/guotianbao/p/12365720.html
Copyright © 2020-2023  润新知