searchusermenu
  • 发布文章
  • 消息中心
点赞
收藏
评论
分享
原创

关于golang interface中T和*T的一个小测试和思考

2023-04-11 11:29:21
21
0

首先

贴一下参考的博客资料。 https://www.cnblogs.com/shijingxiang/articles/12201984.html

背景:

golang中使用interface实现类似于C++或者JAVA 的多态,其中比较复杂的是接口中函数调用者参数可能为T或者*T。 按照官方文档说明,interface中 T 类型只能处理接收者参数为 T, T类型可以调用接收者参数为T或者T。

结论:

interface调用 *T类型调用接收者参数为T的函数时候,做法应该是对 *T类型变量做一个 反操作 类似于下面

package main

import (
   "fmt"
)

type Submit interface {
   work()
}
type Student struct {
   Name string
}

func (s Student) work() {
   fmt.Printf("work: %p\n", &s)
   s.Name += "s"
}
func doWork(sub Submit) {
   sub.work()
   sub.work()
}

func main() {
   stu := &Student{Name: "wyh"}
   fmt.Printf("main: %p\n", stu)
   // 此处传递为Student指针,work函数为Student类型
   fmt.Println(stu)
   doWork(stu)
   fmt.Println(stu)
}
结果为
main: 0xc000048230
&{wyh}
work: 0xc000048260
work: 0xc000048270
&{wyh}
可以看到两次work函数的Student不是同个地址
在以上代码中`sub.work()`函数被调用的时候,我理解编译器做的事情为
var temp Student
temp = *(sub.(&Student)).DeepCopy()
temp.work()

可以多做一个尝试,当work函数为接收者参数为*Student的时候,work函数每一次调用的s打印出来地址都是一致

思考

  • 为什么T类型只能调用接收者参数为T,而不能是*T,反之*T可以调用T和*T。

两个问题,后一个*T可以调用T和*T,原理就是上面猜测。 T类型调用*T会发生什么可能,首先我觉得技术是可以做到,相当于对一个对象寻址应该是没啥问题 但为什么Golang不支持的原因是可能寻址过后,会有人误把它指向另一个其他对象A,假设A又不符合interface定义,会发生内部逻辑错误 这点的推测是从第一种情况中得出,为什么*T调用T的时候,是每次都实例化一个新的对象, 不就是为了安全,而且保持一种约定,*T才能修改对象内容,T不能修改内容。

0条评论
0 / 1000
吴溢豪
5文章数
0粉丝数
吴溢豪
5 文章 | 0 粉丝
原创

关于golang interface中T和*T的一个小测试和思考

2023-04-11 11:29:21
21
0

首先

贴一下参考的博客资料。 https://www.cnblogs.com/shijingxiang/articles/12201984.html

背景:

golang中使用interface实现类似于C++或者JAVA 的多态,其中比较复杂的是接口中函数调用者参数可能为T或者*T。 按照官方文档说明,interface中 T 类型只能处理接收者参数为 T, T类型可以调用接收者参数为T或者T。

结论:

interface调用 *T类型调用接收者参数为T的函数时候,做法应该是对 *T类型变量做一个 反操作 类似于下面

package main

import (
   "fmt"
)

type Submit interface {
   work()
}
type Student struct {
   Name string
}

func (s Student) work() {
   fmt.Printf("work: %p\n", &s)
   s.Name += "s"
}
func doWork(sub Submit) {
   sub.work()
   sub.work()
}

func main() {
   stu := &Student{Name: "wyh"}
   fmt.Printf("main: %p\n", stu)
   // 此处传递为Student指针,work函数为Student类型
   fmt.Println(stu)
   doWork(stu)
   fmt.Println(stu)
}
结果为
main: 0xc000048230
&{wyh}
work: 0xc000048260
work: 0xc000048270
&{wyh}
可以看到两次work函数的Student不是同个地址
在以上代码中`sub.work()`函数被调用的时候,我理解编译器做的事情为
var temp Student
temp = *(sub.(&Student)).DeepCopy()
temp.work()

可以多做一个尝试,当work函数为接收者参数为*Student的时候,work函数每一次调用的s打印出来地址都是一致

思考

  • 为什么T类型只能调用接收者参数为T,而不能是*T,反之*T可以调用T和*T。

两个问题,后一个*T可以调用T和*T,原理就是上面猜测。 T类型调用*T会发生什么可能,首先我觉得技术是可以做到,相当于对一个对象寻址应该是没啥问题 但为什么Golang不支持的原因是可能寻址过后,会有人误把它指向另一个其他对象A,假设A又不符合interface定义,会发生内部逻辑错误 这点的推测是从第一种情况中得出,为什么*T调用T的时候,是每次都实例化一个新的对象, 不就是为了安全,而且保持一种约定,*T才能修改对象内容,T不能修改内容。

文章来自个人专栏
算法
3 文章 | 1 订阅
0条评论
0 / 1000
请输入你的评论
0
0