说说 golang 中的接口和反射

1、接口

1.1 类型

Golang 中的接口是一组方法的签名,是实现多态和反射的基础。

type 接口名 interface {method1(参数列表) 返回值列表method2(参数列表) 返回值列表
}

不同于 Java 语言,使用 implements 关键字显示的实现接口。Golang 接口的实现都是隐式的,只需要实现了接口类型中的所有方法就实现了接口。

func (t 自定义类型) method1(参数列表) (返回值列表) {//方法实现
}func (t 自定义类型) method2(参数列表) (返回值列表) {//方法实现
}

这里来看个示例

type Speaker interface {Speak() string
}type Dog struct{}func (d *Dog) Speak() string {return "Woof!"
}type Cat struct{}func (c *Cat) Speak() string {return "Meow!"
}func PrintSpeak(s Speaker) {fmt.Println(s.Speak())
}func TestSpeaker(t *testing.T) {dog := &Dog{}cat := &Cat{}PrintSpeak(dog)PrintSpeak(cat)
}

在这个例子中,DogCat 类型都实现了 Speaker 接口的 Speak 方法。PrintSpeak 函数接受一个 Speaker 接口类型的参数,因此可以接受任何实现了 Speaker 接口的类型。这展示了多态性的强大之处:不同的类型可以通过相同的接口进行交互。

这里可以再展开下,利用接口多态的特点,服务可以对外提供一个元数据接口,依据传参的不同,返回不同的数据模板。比如,type 字段来做区分,rider 代表返回骑手的相关信息,order 代表返回订单的相关信息,delivery 代表返回运单的相关。服务内部就可以依据 type 的不同,来分别封装实现。

另外,Golang 接口不能包含任何变量,且允许为空。空接口 interface{} 没有任何方法,所以所有类型都实现了空接口。

func TestGeneric(t *testing.T) {var values []interface{}values = append(values, 42)values = append(values, "hello")values = append(values, 3.14)for _, value := range values {fmt.Println(value)}
}

上述代码展示了如何使用空接口 (interface{}) 来存储不同类型的值,并通过循环遍历这些值进行打印,仔细看是不是像泛型?

因为接口在定义一组方法时没有对实现的接收者做限制,所以在这一节的最后,来探讨下结构体实现接口和结构体指针实现接口的不同。

  • 示例一
type Animal interface {Say()
}type Cat struct{}// 接收者类型为结构体指针
func (c *Cat) Say() {fmt.Println("miu")
}func TestStruct(t *testing.T) {var animal Animal = &Cat{}animal.Say()
}

程序正常运行。

  • 示例二
type Animal interface {Say()
}type Cat struct{}// 接收者类型为结构体
func (c Cat) Say() {fmt.Println("miu")
}func TestStruct(t *testing.T) {var animal Animal = &Cat{}animal.Say()
}

程序正常运行。

  • 示例三
type Animal interface {Say()
}type Cat struct{}// 接收者类型为结构体
func (c Cat) Say() {fmt.Println("miu")
}func TestStruct(t *testing.T) {var animal Animal = Cat{}animal.Say()
}

程序正常运行。

  • 示例四
type Animal interface {Say()
}type Cat struct{}// 接收者类型为结构体指针
func (c *Cat) Say() {fmt.Println("miu")
}func TestStruct(t *testing.T) {var animal Animal = Cat{}animal.Say()
}

运行失败,输出

cannot use Cat{} (value of type Cat) as type Animal in variable declaration:Cat does not implement Animal (Say method has pointer receiver)

编译器提示 Cat 没有实现 Animal 接口,Say 方法接受的是指针。

针对上述四个示例汇总如下

结构体实现接口结构体指针实现接口
结构体初始化变量通过不通过
结构体指针初始化变量通过通过

为什么会出现这种情况呢?我们知道 Golang 中传递参数都是值传递

  • 对于 &Cat{} 来说,这意味着拷贝一个新的 &Cat{} 指针,不过这个指针与原来的指针指向一个相同并且唯一的结构体,所以编译器可以隐式的对变量解引用(dereference)获取指针指向的结构体
  • 而对于 Cat{} 来说,这意味着 Say 方法会接受一个全新的 Cat{},因为方法的参数是 *Cat,编译器不会无中生有创建一个新的指针;即使编译器可以创建新指针,这个指针指向的也不是最初调用该方法的结构体;

这里可以看出,当接受者为结构体时,那么在方法调用的时候需要传值,拷贝参数,这里会有性能损失,因此建议在实际项目中,接受者使用结构体指针来实现。

1.2 数据结构

golang 版本 1.19.12

Golang 中有两种略微不同的接口,一种是带有一组方法的接口,另一种是不带任何方法的接口,下面就分别介绍下其底层实现。

1.2.1 空接口

空接口的实现如下

type EmptyInterface {}

其底层的数据结构如下

// runtime/runtime2.go
type eface struct {_type *_typedata unsafe.Pointer
}

eface 的结构体由两个属性构成,一个是类型信息 _type,一个是数据信息 data,占 16 个字节。

  • _type 属性,存放的是类型、方法等信息。
// Needs to be in sync with ../cmd/link/internal/ld/decodesym.go:/^func.commonsize,
// ../cmd/compile/internal/reflectdata/reflect.go:/^func.dcommontype and
// ../reflect/type.go:/^type.rtype.
// ../internal/reflectlite/type.go:/^type.rtype.
type _type struct {size       uintptr // 类型占用内存大小ptrdata    uintptr // 包含所有指针的内存前缀大小hash       uint32  // 类型 hash,用于比较两个类型是否相等tflag      tflag   // 标记位,主要用于反射align      uint8   // 对齐字节信息fieldAlign uint8   // 当前结构字段的对齐字节数kind       uint8   // 基础类型枚举值equal func(unsafe.Pointer, unsafe.Pointer) bool // 比较两个形参对应对象的类型是否相等gcdata    *byte    // GC 类型的数据str       nameOff  // 类型名称字符串在二进制文件段中的偏移量ptrToThis typeOff  // 类型元信息指针在二进制文件段中的偏移量
}

其中可以关注下 kind,这个字段描述的是如何解析基础类型。在 Golang 中,基础类型是一个枚举常量,有 26 个基础类型,如下。枚举值通过 kindMask 取出特殊标记位。

// runtime/typekind.go
const (kindBool = 1 + iotakindIntkindInt8kindInt16kindInt32kindInt64kindUintkindUint8kindUint16kindUint32kindUint64kindUintptrkindFloat32kindFloat64kindComplex64kindComplex128kindArraykindChankindFunckindInterfacekindMapkindPtrkindSlicekindStringkindStructkindUnsafePointerkindDirectIface = 1 << 5kindGCProg      = 1 << 6kindMask        = (1 << 5) - 1
)

这里再做个简单的展开, kindMask 的值为 31,对应的二进制为 00011111,也就是低五位都为 1,再看下 Golang 中有 26 个基础类型,也就是都比 kindMask 值要小,这时利用位与运算的特性(如果两个对应位都是 1,则结果位为 1,否则为 0),二者做与运算可以获取对应类型的种类信息。

  • data 属性,指向原始数据的指针,是一个 unsafe.Pointer 类型

下面用个示例演示空接口 eface 数据到底是如何存储的。

type nameOff int32 // offset to a name
type typeOff int32 // offset to an *rtype
type tflag uint8type eface struct {_type *_typedata  unsafe.Pointer
}type _type struct {size       uintptrptrdata    uintptr // size of memory prefix holding all pointershash       uint32tflag      tflagalign      uint8fieldAlign uint8kind       uint8// function for comparing objects of this type// (ptr to object A, ptr to object B) -> ==?equal func(unsafe.Pointer, unsafe.Pointer) bool// gcdata stores the GC type data for the garbage collector.// If the KindGCProg bit is set in kind, gcdata is a GC program.// Otherwise it is a ptrmask bitmap. See mbitmap.go for details.gcdata    *bytestr       nameOffptrToThis typeOff
}func TestEFace(t *testing.T) {var i interface{} = 3.14e := (*eface)(unsafe.Pointer(&i))fmt.Println(e)fmt.Println(e._type)fmt.Println(e.data)}

打印输出一目了然。

&{0x1049dc640 0x1049ce618}
&{8 0 2472095124 7 8 8 14 0x1048bafd0 0x1049ce678 7052 34048}
0x1049ce618

从上述 eface 结构的两个属性可以推断出,Golang 的任意类型都可以转换成 interface{}

1.2.2 非空接口

空接口的实现如下

type NoEmptyInterface {Say()
}

其底层的数据结构如下

type iface struct {tab  *itabdata unsafe.Pointer
}

iface 结构体也是占 16 个字节,这里的 data 属性和空接口的 eface 里的 data 作用相同,这里就不做赘述。下面来看看 tab 属性。

// layout of Itab known to compilers
// allocated in non-garbage-collected memory
// Needs to be in sync with
// ../cmd/compile/internal/reflectdata/reflect.go:/^func.WriteTabs.
type itab struct {inter *interfacetype // 存的是 interface 自己的静态类型_type *_type // 存的是 interface 对应具体对象的类型hash  uint32 // 是对 _type.hash 的拷贝_     [4]bytefun   [1]uintptr // 是一个函数指针,它指向的是具体类型的函数方法
}

itab 结构体是非空接口的核心组成部分,占 32 字节,着重看下 inter_type 属性。

type imethod struct {name nameOffityp typeOff
}type interfacetype struct {typ     _type // 类型元信息pkgpath name  // 包路径和描述信息等等mhdr    []imethod // 方法
}

inter 存储的是非空接口自己类型相关数据,因为 Golang 中函数方法是以包为单位隔离的。所以 interfacetype 除了保存 _type 还需要保存包路径等描述信息。mhdr 存的是各个 interface 函数方法在段内的偏移值 offset,知道偏移值以后才方便调用。

_type 在上一节空接口已详细介绍过,存储的是接口类型的元信息,这里就不展开。

这里还要说下 tabfun 属性,存储的是指向实现非空接口类型的方法数组。

type iface struct {tab  *itabdata unsafe.Pointer
}type itab struct {inter *interfacetype_type *_typehash  uint32 // copy of _type.hash. Used for type switches._     [4]bytefun   [1]uintptr // variable sized. fun[0]==0 means _type does not implement inter.
}type imethod struct {name nameOffityp typeOff
}type interfacetype struct {typ     _typepkgpath namemhdr    []imethod
}type name struct {bytes *byte
}type _type struct {size       uintptrptrdata    uintptr // size of memory prefix holding all pointershash       uint32tflag      tflagalign      uint8fieldAlign uint8kind       uint8// function for comparing objects of this type// (ptr to object A, ptr to object B) -> ==?equal func(unsafe.Pointer, unsafe.Pointer) bool// gcdata stores the GC type data for the garbage collector.// If the KindGCProg bit is set in kind, gcdata is a GC program.// Otherwise it is a ptrmask bitmap. See mbitmap.go for details.gcdata    *bytestr       nameOffptrToThis typeOff
}type People interface{Say()
}type Student struct {}func (s *Student) Say() {fmt.Println("hello world")
}func TestNoEmptyInterface(t *testing.T) {s := &Student{}var i People = snoEmpty := (*iface)(unsafe.Pointer(&i))fmt.Println(noEmpty)fmt.Println(noEmpty.tab)fmt.Println(noEmpty.tab.inter)fmt.Println(noEmpty.tab._type)fmt.Println(noEmpty.tab.fun)fmt.Println(noEmpty.data)}

打开 Goland 的调试模式,可以很清晰的看到非空接口内部字段都是如何存储的。

type Student struct{}func (stu *Student) Show() {}func live() People {var stu *Studentreturn stu
}func TestNil(t *testing.T) {stu := live()if stu == nil {fmt.Println("nil")} else {fmt.Println("not nil")}}

看完这一节,上面输出应该就简单了吧,输出 not nil。至于原因嘛,在运行时,非空接口只是 datanil,但是 tab 可不为 nil

这里再做个展开,简单说说动态派发。

Gointerface 可以动态派发方法,实现类似面向对象语言中的多态的特性。

func TestDynamic(t *testing.T) {var stu People = &Student{}stu.Show()
}

这里为了正常实现 stu.Show() 方法调用,需要构建 iface 结构,之后再依据 *tab 里面存的 fun 指针做一次寻址,接着才能调用。

func TestDynamic(t *testing.T) {var stu := &Student{}stu.Show()
}

这里直接调用结构体的方法,少了构建 iface 结构以及寻址的时间,性能应该比动态派发要好。不过,指针实现的动态派发造成的性能损失非常小,相对于一些复杂逻辑的处理函数,这点性能损失几乎可以忽略不计。

2、反射

Golang 中的反射是用标准库中的 reflect 包实现,reflect 包实现了 runtime (运行时)的反射能力,能够让程序操作不同的对象。

reflect 包中有两个非常重要的函数:

  • reflect.TypeOf 能获取类型信息;
  • reflect.ValueOf 能获取数据的运行时表示;

2.1 TypeOf

先来看看 TypeOf 函数

// reflect/type.go
// TypeOf returns the reflection Type that represents the dynamic type of i.
// If i is a nil interface value, TypeOf returns nil.
func TypeOf(i any) Type {eface := *(*emptyInterface)(unsafe.Pointer(&i))return toType(eface.typ)
}// toType converts from a *rtype to a Type that can be returned
// to the client of package reflect. In gc, the only concern is that
// a nil *rtype must be replaced by a nil Type, but in gccgo this
// function takes care of ensuring that multiple *rtype for the same
// type are coalesced into a single Type.
func toType(t *rtype) Type {if t == nil {return nil}return t
}

其中入参 i 的类型 any 就是 interface{} 的别名,anyGolang 1.18 中引入,表示任意类型。

type any = interface{}

TypeOf 函数实现很简单,只是将一个 interface{} 变量转换成了内部的 emptyInterface 表示,然后从中获取相应的类型信息。里面有个强制转换,把 any 转成了 emptyInterface。下面来看看 emptyInterface 的数据结构:

// emptyInterface is the header for an interface{} value.
type emptyInterface struct {typ  *rtypeword unsafe.Pointer
}// rtype is the common implementation of most values.
// It is embedded in other struct types.
//
// rtype must be kept in sync with ../runtime/type.go:/^type._type.
type rtype struct {size       uintptrptrdata    uintptr // number of bytes in the type that can contain pointershash       uint32  // hash of type; avoids computation in hash tablestflag      tflag   // extra type information flagsalign      uint8   // alignment of variable with this typefieldAlign uint8   // alignment of struct field with this typekind       uint8   // enumeration for C// function for comparing objects of this type// (ptr to object A, ptr to object B) -> ==?equal     func(unsafe.Pointer, unsafe.Pointer) boolgcdata    *byte   // garbage collection datastr       nameOff // string formptrToThis typeOff // type for pointer to this type, may be zero
}

仔细看看 emptyInterface 结构,发现其属性和上一章节的空接口是一模一样的,只是名称改了下, rtype 用于表示变量的类型, word 指向内部封装的数据。

TypeOf 参数 i any 中可以看出,入参是两种类型

  • 一种是具体类型变量,TypeOf() 返回的具体类型信息
  • 一种是 interface 类型变量
    • 如果 i 绑定了具体类型对象实例,返回的是 i 绑定具体类型的动态类型信息;
    • 如果 i 没有绑定任何具体的类型对象实例,返回的是接口自身的静态类型信息。
type Forest interface{}type Tree struct{}func TestBasicType(t *testing.T) {ifa := &Tree{}rfa := reflect.TypeOf(ifa)fmt.Println("第一组输出:", rfa.Elem().Name(), rfa.Elem().Kind().String())var ifb Forest = &Tree{}ifc := new(Forest)rfb := reflect.TypeOf(ifb)rfc := reflect.TypeOf(ifc)fmt.Println("第二组输出:", rfb.Elem().Name(), rfb.Elem().Kind().String())fmt.Println("第二组输出:", rfc.Elem().Name(), rfc.Elem().Kind().String())
}

输出

第一组输出: Tree struct
第二组输出: Tree struct
第二组输出: Forest interface
  • 第一组输出中 ifa 是具体的类型,所以返回本身的类型 Tree,对应的 kindstruct
  • 第二组输出中
    • ifbinterface 类型变量,绑定具体的类型对象实例,所以返回的是绑定的具体类型 Forest,对应的 kindstruct
    • ifcinterface 类型变量,且没有绑定任何具体的类型对象实例,所以返回的是本身的类型 Forest,对应的 kindinterface

2.2 ValueOf

再来看看 ValueOf 函数

// reflect/type.go
// ValueOf returns a new Value initialized to the concrete value
// stored in the interface i. ValueOf(nil) returns the zero Value.
func ValueOf(i any) Value {if i == nil {return Value{}}// TODO: Maybe allow contents of a Value to live on the stack.// For now we make the contents always escape to the heap. It// makes life easier in a few places (see chanrecv/mapassign// comment below).escapes(i)return unpackEface(i)
}// Dummy annotation marking that the value x escapes,
// for use in cases where the reflect code is so clever that
// the compiler cannot follow.
func escapes(x any) {if dummy.b {dummy.x = x}
}var dummy struct {b boolx any
}// unpackEface converts the empty interface i to a Value.
func unpackEface(i any) Value {e := (*emptyInterface)(unsafe.Pointer(&i))// NOTE: don't read e.word until we know whether it is really a pointer or not.t := e.typif t == nil {return Value{}}f := flag(t.Kind())if ifaceIndir(t) {f |= flagIndir}return Value{t, e.word, f}
}

ValueOf 实现也很简单,若 inil 返回零值 Value{}。否则就先调用 escapes(i) 确保值逃逸到堆上,然后调用 reflect.unpackEface ,先把 i interface{} 转换成 emptyInterface,然后将具体类型和指针包装成 reflect.Value 结构体后返回。

这里对内存逃逸做个简单的说明。在 Golang 中,内存逃逸通常发生在以下情况:

1.变量的地址被返回给调用者。
2.变量的地址被赋值给全局变量或者其他包级变量。
3.变量的地址被传递给不透明的函数或方法(例如接口方法)。

上面的 dump 是个全局变量,当 x 赋值给全部变量的 x 属性时就会发生逃逸。其实,在变量 i 传给函数 escapes 时就已经发生了逃逸(见上述第三种情况),笔者在本机环境中(Golang 版本为 1.19.12)测试也是如此:

package _0240619import "testing"func escapes(x any) {if dummy.b {dummy.x = x}
}var dummy struct {b boolx any
}func TestEscapes(t *testing.T) {i := 11escapes(i)
}

执行 go tool compile -m escapes_test.go ,输出如下

escapes_test.go:5:6: can inline escapes
escapes_test.go:16:6: can inline TestEscapes
escapes_test.go:19:9: inlining call to escapes
escapes_test.go:5:14: leaking param: x
escapes_test.go:16:18: t does not escape
escapes_test.go:19:9: i escapes to heap

这里的 19 行就是 escapes(i),也就是传参的时候由于此函数入参是个 interface{} 因此发生了内存逃逸。

那什么时候仍然需要 escapes 函数?如果在某些特定情况下,编译器无法自动判定变量需要逃逸到堆上,而又需要强制变量逃逸,那么 escapes 函数仍然是有用的。它可以明确告诉编译器这个变量需要在堆上分配。

那为什么要在 ValueOf 中把变量的内存逃逸到堆上呢?源码上写的原因是 这样标记是为了防止反射代码写的过于高级,以至于编译器跟不上了。

这里我理解为 ValueOf 通常用于反射操作,在反射中我们经常需要确保被反射的对象在堆上,以便反射包能够安全地访问和修改这些对象。例如:

  • 反射可能需要长时间持有对象的引用。
  • 反射可能会传递对象到其他地方(例如,作为返回值)。
  • 反射可能会在不同的 Goroutine 之间传递对象。

2.3 反射三定律

Golang 开派祖师之一 Rob Pike2011 年写的 The Laws of Reflection,提出了反射三大定律

  1. 反射可以从接口值到反射对象
  2. 反射可以从反射对象中获得接口值
  3. 要修改反射对象,其值必须可设置

2.3.1 反射可以从接口值得到反射对象

反射第一定律即通过一个接口值,我们可以获取其对应的反射对象。可以通过 reflect.TypeOfreflect.ValueOf 函数来实现。因为二者的入参都是 interface{},传参的时候会发生类型转换。

func TestReflectFirstLaw(t *testing.T) {var x float64 = 3.4tp := reflect.TypeOf(x) // 得到类型信息 reflect.Typev := reflect.ValueOf(x) // 得到值信息 reflect.Valuefmt.Println("type:", tp) // 输出 float64fmt.Println("value:", v) // 3.4}

2.3.2 反射可以从反射对象中获得接口值

即通过一个反射对象,我们可以还原出其对应的接口值。可以通过 reflect.ValueInterface 方法来实现。

不过调用 reflect.Value.Interface 方法只能获得 interface{} 类型的变量,如果想要将其还原成最原始的状态还需要经过如下所示的显式类型转换:

func TestReflectSecondLaw(t *testing.T) {var x float64 = 3.4v := reflect.ValueOf(x)y := v.Interface().(float64) // 将反射值转回为接口值fmt.Println(y) // 3.4
}

仔细看,发现第一定律和第二定律是互为逆向的过程:

  • 从接口值得到反射对象:
    • 从基本类型到接口类型的类型转换(这里是隐式转换);
    • 从接口类型到反射对象的转换;
  • 从反射对象的到接口值:
    • 反射对象转换成接口类型;
    • 通过显式类型转换变成原始类型;

2.3.3 要修改反射对象,其值必须可设置

如果我们想要更新一个 reflect.Value,那么它持有的值一定是可以被更新的。

func TestReflectThirdLaw(t *testing.T) {f := 3.14v := reflect.ValueOf(f)v.SetFloat(6.52)
}

运行后报如下错误

panic: reflect: reflect.Value.SetFloat using unaddressable value [recovered]panic: reflect: reflect.Value.SetFloat using unaddressable value

这里给的提示信息是使用了不可寻址(unaddressable)的 Value。

我们知道 Golang 的函数的参数都是值传递的,这里 reflect.ValueOf(f) 传值后,会拷贝一个 f 的副本,与原来的 f 就没关系了,此时再改 f 的值就会报不可寻址。

修复也简单,传指针即可,虽然是值拷贝,但是指向的都是同一块内存地址。

func TestReflectThirdLaw(t *testing.T) {f := 3.14v := reflect.ValueOf(f)if !v.CanSet() {v = reflect.ValueOf(&f)v = v.Elem()}v.SetFloat(6)fmt.Println(v)}

2.4 实例

首先通过一个综合的示例,来把上面列举的 TypeOfValueOf 给串起来。

type Monster struct {Name  string `json:"name"`Age   int    `json:"monster_age"`Score float32Sex   string
}func (m *Monster) Print() {fmt.Println("---start---")fmt.Println(m)fmt.Println("---end---")
}func (m *Monster) GetSum(n1, n2 int) int {return n1 + n2
}func (m *Monster) Set(name string, age int, score float32, sex string) {m.Name = namem.Age = agem.Score = scorem.Sex = sex
}func ProcStruct(a interface{}) {typ := reflect.TypeOf(a)val := reflect.ValueOf(a)valMethod := valif typ.Kind() == reflect.Ptr {typ = typ.Elem()}if typ.Kind() != reflect.Struct {fmt.Println("expect struct")return}if val.Kind() == reflect.Ptr {val = val.Elem()}num := val.NumField()fmt.Printf("struct has %d field\n", num)for i := 0; i < num; i++ {fmt.Printf("Field %d: 属性为 = %v\n", i, typ.Field(i).Name)fmt.Printf("Field %d: 值为 = %v\n", i, val.Field(i))tagVal := typ.Field(i).Tag.Get("json")if tagVal != "" {fmt.Printf("Field %d: tag 为 %v\n", i, tagVal)}}numOfMethod := valMethod.NumMethod()fmt.Printf("struct has %d methods\n", numOfMethod)valMethod.Method(1).Call(nil)var params []reflect.Valueparams = append(params, reflect.ValueOf(10))params = append(params, reflect.ValueOf(40))res := valMethod.Method(0).Call(params)fmt.Println("res = ", res[0].Int())}func TestReflectDemo(t *testing.T) {monster := &Monster{Name:  "Cat",Age:   400,Score: 30.8,Sex:   "10",}ProcStruct(monster)}

打印输出

struct has 4 field
Field 0: 属性为 = Name
Field 0: 值为 = Cat
Field 0: tag 为 name
Field 1: 属性为 = Age
Field 1: 值为 = 400
Field 1: tag 为 monster_age
Field 2: 属性为 = Score
Field 2: 值为 = 30.8
Field 3: 属性为 = Sex
Field 3: 值为 = 10
struct has 3 methods
---start---
&{Cat 400 30.8 10}
---end---
res =  50

这里需要关注下:

  • NumField(),获取结构体字段的数量
  • NumMethod, 获取 reflect.Value 可以访问的方法数量,这包括值接收者和指针接收者的方法
    • 值接收器:方法定义在值类型上。例如:func (m Monster) Print()
    • 指针接收器:方法定义在指针类型上。例如:func (m *Monster) Print()

来看看下面这段代码

type Foo struct {Name string
}func (f *Foo) Method1() {fmt.Println("Method1 called")
}func TestNum(t *testing.T) {f := &Foo{}v := reflect.ValueOf(f)fmt.Println(v)fmt.Println(v.Elem())fmt.Println(v.Elem().NumField())fmt.Println(v.NumMethod())
}

打印输出

&{}
{}
1
1

再结合上面列出的关注点,就很清楚二者要获取数量的注意点了。

  • NumField(),针对的是结构体本身的字段数量,若接收器(receiver)是结构体指针,需要解引用指针,获取指针指向的 reflect.Value,也就是结构体
  • NumMethod(),用于获取某个 reflect.Value 所代表的类型的方法数量,即使类型为指针,也无需使用 Elem() 进行解引用。

其次,再来个依据反射动态调用方法

type DynamicStruct struct {Name stringAge  intAddr string
}func (d *DynamicStruct) PrintName(ctx context.Context) error {fmt.Println(d.Name)return nil
}func (d *DynamicStruct) PrintAge(ctx context.Context) error {fmt.Println(d.Age)return nil
}func (d *DynamicStruct) PrintAddr(ctx context.Context) error {fmt.Println(d.Addr)return nil
}func TestDynamic(t *testing.T) {dynamic := &DynamicStruct{Name: "molaifeng",Age:  18,Addr: "beijing",}ctx := context.Background(){method, exits := dynamic.GetMethod("Name")if !exits {return}_ = method(ctx)}{method, exits := dynamic.GetMethod("hobby")if !exits {return}_ = method(ctx)}}func (d *DynamicStruct) GetMethod(name string) (func(ctx context.Context) error, bool) {methodName := "Print" + namemethod := reflect.ValueOf(d).MethodByName(methodName)if !method.IsValid() {fmt.Println(fmt.Printf("%v not exist", methodName))return nil, false}return method.Interface().(func(ctx context.Context) error), true
}

打印输出

molaifeng
Printhobby not exist20 <nil>

这里主要用到了 MethodByName 获取反射对象的方法,有了前面例子打底,这里就不做过多介绍。另外一个就是 Interface() 方法了,这个方法可以获取反射对象,但是呢,要还原成原本的类型,还需显示的转换,于是就有下面这个

method.Interface().(func(ctx context.Context) error), true

其实这个很好理解,看看具体 *DynamicStruct 具体绑定的方法,返回值是不是就是上面括号里的。

再来个简化版的

func TestInt(t *testing.T) {num := 99v := reflect.ValueOf(num)i := v.Interface().(int)fmt.Println(i)}

最后,来个校验参数的例子

type Params struct {Name *stringAge  *intAddr *string
}func TestValidParam(t *testing.T) {requiredParam := map[string]bool{"Name": true,"Age":  true,}params := &Params{}err := CheckParams(params, requiredParam)if err != nil {fmt.Println(err)return}}func CheckParams(params *Params, requiredParam map[string]bool) error {val := reflect.ValueOf(params).Elem()for fieldName := range requiredParam {field := val.FieldByName(fieldName)if isFieldNil(field) {return fmt.Errorf("%v is required,actual nil", fieldName)}}return nil}func isFieldNil(v reflect.Value) bool {k := v.Kind()switch k {case reflect.Slice, reflect.Map, reflect.Ptr:return v.IsNil()default:return false}
}

打印如下

Name is required,actual nil

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/32620.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【服务器06】之【如何不开外网连接GitHub】

登录GitHub官网 GitHub: Let’s build from here GitHub 注册账号 登录账号 输入一个自定义名字&#xff0c;点击创建存储库就可以了 首先 如何在不开外网的条件下使用GitHub 第一步 下载安装Steam(Watt TooklKit) 区分一下如何查看哪个官网&#xff08;没有百度广告就是…

实时导航 C语言

实现实时导航的一种方法是使用C语言和一些外部库&#xff0c;如SDL&#xff08;Simple DirectMedia Layer&#xff09;来处理图形界面和输入&#xff0c;以及OpenGL进行图形渲染。以下是一个简单的例子&#xff0c;展示了如何使用C语言和SDL库来创建一个简单的实时导航应用。 …

Package libcudnn8 is not installed.的问题

ubantu20.04&#xff0c;cuda11.2安装cuda&#xff0c; cudann . 安装CUDNN时&#xff0c;如果选择 Deb安装方法&#xff0c; 有三个安装包要安装&#xff0c;必须先安装runtime版本&#xff0c;再安装developer版本&#xff0c;不然就会出现以下问题&#xff1a; dpkg: depe…

如何在Android中实现多线程与线程池?

目录 一、Android介绍二、什么是多线程三、什么是线程池四、如何在Android中实现多线程与线程池 一、Android介绍 Android是一种基于Linux内核的开源操作系统&#xff0c;由Google公司领导开发。它最初于2007年发布&#xff0c;旨在为移动设备提供一种统一、可扩展的操作系统。…

UltraEditUEStudio软件安装包下载及安装教程

​根据软件大数据显示提供预定义的或使用者创建的编辑“环境”&#xff0c;能记住 UltraEdit 的所有可停靠窗口、工具栏等的状态。实际上我们可以这样讲HTML 工具栏&#xff0c;对常用的 HTML 功能作了预配置;文件加密/解密;多字节和集成的 IME。根据使用者情况表明Git Editor&…

【Linux基础IO】磁盘的结构、文件系统与inode、软硬链接

目录 磁盘的物理存储结构 磁盘的逻辑结构 文件系统与inode 硬链接 软链接 关于目录文件 磁盘的物理存储结构 磁盘是计算机中唯一的机械硬件设备&#xff0c;由磁头&#xff0c;盘面&#xff0c;磁道&#xff0c;扇区这四部分对磁盘上的数据进行增删查改操作。盘面上存储的…

面向对象修炼手册(二)(消息与继承)(Java宝典)

&#x1f308; 个人主页&#xff1a;十二月的猫-CSDN博客 &#x1f525; 系列专栏&#xff1a; &#x1f3c0;面向对象修炼手册 &#x1f4aa;&#x1f3fb; 十二月的寒冬阻挡不了春天的脚步&#xff0c;十二点的黑夜遮蔽不住黎明的曙光 目录 前言 消息传递 1 基本概念 1.…

App推广告别邀请码,Xinstall助您一键触达海量用户!

在移动互联网高速发展的今天&#xff0c;App的推广与运营已成为每个开发者都必须面对的问题。然而&#xff0c;随着互联网流量的日益分散和用户需求的不断变化&#xff0c;传统的App推广方式已经难以满足现代市场的需求。尤其是在获取用户时&#xff0c;很多开发者还在采用传统…

动手学深度学习(Pytorch版)代码实践 -卷积神经网络-26网络中的网络NiN

26网络中的网络NiN import torch from torch import nn import liliPytorch as lp import matplotlib.pyplot as plt# 定义一个NiN块 def nin_block(in_channels, out_channels, kernel_size, strides, padding):return nn.Sequential(# 传统的卷积层nn.Conv2d(in_channels, ou…

图的学习.

目录 一、图的基本概念 1.1图的种类 1.2顶点的度、入度和出度 1.3边的权和网 1.4路径、路径长度和回路 二、图的存储结构 2.1邻接矩阵法 2.2邻接表法 2.3十字链表 2.4邻接多重表 三、图的遍历 3.1广度优先搜索 3.2深度优先搜索 四、图的应用 4.1最小生成树 4.1.…

VSCode 安装Remote-SSH

1、打开扩展商店安装Remote-SSH 快捷键&#xff1a;CtrlShiftX 2、配置ssh连接 打开命令面板&#xff08;CtrlShiftP&#xff09; 输入"Remote-SSH: Connect to Host"并选择。 输入你的Ubuntu服务器的IP地址或主机名。 3、连接到ubuntu服务器 如果是第一次连接&…

FPGA结构相关简介

一、芯片分类 ​FPGA属于数字芯片的一种&#xff0c;下面是根据世界半导体贸易统计协会WSTS的一个半导体分类&#xff0c;可以看到FPGA所属的类别。 二、FPGA的发展史 ​下图为FPGA的发展历史 三、FPGA的结构分类 下面是从三个角度进行划分 四、参考资料 《FPGA原理与结构》—…

【课程总结】Day10:卷积网络的基本组件

前言 由于接下来的课程内容将围绕计算机视觉展开&#xff0c;其中接触最多的内容是卷积、卷积神经网络等…因此&#xff0c;本篇内容将从卷积入手&#xff0c;梳理理解&#xff1a;卷积的意义、卷积在图像处理中的作用以及卷积神经网络的概念&#xff0c;最后利用pytorch搭建一…

1169 正方形

简单- 时间限制&#xff1a; 1000MS内存限制&#xff1a; 256MB分数&#xff1a;100OI排行榜得分&#xff1a;10(0.1*分数2*难度) 考点&#xff1a;选择结构 描述 有一个正方形&#xff0c;四个角的坐标分别是&#xff08;1&#xff0c;-1&#xff09;&#xff0c;&#xff…

proactor模式

Proactor模式是一种异步I/O的设计模式&#xff0c;它允许程序直接发起一个异步I/O操作并立即返回&#xff0c;而不需要等待该操作完成。一旦I/O操作实际完成&#xff0c;系统会通知相应的完成处理程序&#xff08;Completion Handler&#xff09;&#xff0c;该处理程序随后执行…

构建未来应用的核心,云原生技术栈解析

&#x1f407;明明跟你说过&#xff1a;个人主页 &#x1f3c5;个人专栏&#xff1a;《未来已来&#xff1a;云原生之旅》&#x1f3c5; &#x1f516;行路有良友&#xff0c;便是天堂&#x1f516; 目录 一、云原生技术栈 1、容器和容器编排 1.1 Docker 1.2 Kubernete…

ULTRAINTERACT 数据集与 EURUS 模型:推动开源大型语言模型在推理领域的新进展

在人工智能的浪潮中&#xff0c;大型语言模型&#xff08;LLMs&#xff09;已经成为推动自然语言处理技术发展的关键力量。它们在理解、生成语言以及执行复杂任务方面展现出了巨大的潜力。然而&#xff0c;尽管在特定领域内取得了显著进展&#xff0c;现有的开源LLMs在处理多样…

js中数组去重的方法

数组去重有的方法&#xff1a; 1、用 ES6 中的 Set 数据结构&#xff1a; let array [1, 2, 2, 3, 4, 4, 5]; let uniqueArray [...new Set(array)]; console.log(uniqueArray); // [1, 2, 3, 4, 5]2、这种方法利用了 Set 对象不允许包含重复项的特性。你也可以使用 ES5 的…

Vue3+ElementPlus+pinia 小案例

Vue3ElementPluspinia 小案例 1、初始化项目 使用脚手架快速创建Vue3应用&#xff1a;https://cli.vuejs.org/zh/ 脚手架自动整合了vue-router路由、ts、前端工程化等库&#xff1b; 安装脚手架工具 npm install -g vue/cli检测安装是否成功 vue -V创建项目&#xff1a; …

数据结构之B树:深入了解与应用

目录 1. B树的基本概念 1.1 B树的定义 1.2 B树的性质 1.3 B树的阶 2. B树的结构 2.1 节点结构 2.2 节点分裂 2.3 节点合并 3. B树的基本操作 3.1 搜索 3.2 插入 3.3 删除 4. B树的应用 4.1 数据库索引 4.2 文件系统 4.3 内存管理 5. B树的优势和局限 5.1 优势…