编程并不是一个机械性的工作,而是需要有思考,有创新的工作,语法是固定的,但解决问题的思路则是依靠人的思维,这就需要我们坚持学习和更新自己的知识。今天编程网就整理分享《如何使用反射获取私有类型?》,文章讲解的知识点主要包括,如果你对Golang方面的知识点感兴趣,就不要错过编程网,在这可以对大家的知识积累有所帮助,助力开发能力的提升。
问题内容假设有人使用 client, _ := rpc.dial('tcp', 'example.com:port')
初始化了一个 rpc 客户端,并且只为您提供 rpc 客户端实例。出于某种原因,您希望从客户端实例获取地址 example.com
。但是,由于客户端类型是高度抽象的,因此您无法直接获取提供 remoteaddr()
方法的 net.conn
实例。在这种情况下,reflect 包可能会有所帮助。
让我们从以下尝试开始:
func getremoteaddr(client *rpc.client) net.addr {
codec := reflect.indirect(rpclient).fieldbyname("codec")
connv := codec.fieldbyname("rwc")
conn := connv.interface().(net.conn)
return conn.remoteaddr()
}
但是,第二行立即跳出错误:
panic: reflect: call of reflect.value.fieldbyname on interface value
这是合理的,因为 codec
被声明为 clientcodec
类型,它只是一个具有四个方法签名的接口。但是,从代码中我们知道codec
的详细类型可能是未导出的结构体gobclientcodec
。不幸的是,go 编译器不够智能,无法深入追踪并找出实际类型。因此,有没有一种方法可以使用反射包告诉编译器我确信 codec
的类型是未导出的 gobclientcodec
?如果这是不可能的,我可能必须计算地址偏移量并使用不安全的指针,这是最后的选择。
附注reflect 方法中的默认 type()
方法将不起作用,因为 codec
被声明为 clientcodec
并通过在 gobclientcodec
方法中传递 gobclientcodec
结构的地址进行初始化,如下所示:
func NewClient(conn io.ReadWriteCloser) *Client {
encBuf := bufio.NewWriter(conn)
client := &gobClientCodec{conn, gob.NewDecoder(conn), gob.NewEncoder(encBuf), encBuf}
return NewClientWithCodec(client)
}
此外,reflect.typeof(rpc.gobclientcodec)
将无法正常工作,因为编译器会拒绝未导出的类型使用。
正确答案
使用.Elem()
获取接口中包含的值。
elem 返回接口 v 包含的值或者 指针 v 指向。如果 v 的 kind 不是 interface 或 ptr,它会发生恐慌。它 如果 v 为零,则返回零值。
对于您的特定用例,您还需要使用 unsafe
包来访问未导出的字段。使用之前请务必阅读 unsafe 的文档。
func getRemoteAddr(client *rpc.Client) net.Addr {
rv := reflect.ValueOf(client)
rv = rv.Elem() // deref *rpc.Client
rv = rv.FieldByName("codec") // get "codec" field from rpc.Client
rv = rv.Elem() // get the value that ClientCodec contains
rv = rv.Elem() // deref *gobClientCodec
rv = rv.FieldByName("rwc") // get "rwc" field from gobClientCodec
conn := *(*net.Conn)(unsafe.Pointer(rv.UnsafeAddr()))
return conn.RemoteAddr()
}
https://go.dev/play/p/cHHdwQJ7DE7
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《如何使用反射获取私有类型?》文章吧,也可关注编程网公众号了解相关技术文章。