十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
这篇文章主要讲解了“Go语言中结构体方法副本传参与指针传参的区别”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Go语言中结构体方法副本传参与指针传参的区别”吧!
为汤旺等地区用户提供了全套网页设计制作服务,及汤旺网站建设行业解决方案。主营业务为网站建设、网站制作、汤旺网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!GO语言结构体方法跟结构体指针方法的区别
首先,我定了三个接口、一个结构和三个方法:
type DeptModeA interface { Name() string SetName(name string) } type DeptModeB interface { Relocate(building string, floor uint8) } type Dept struct { name string building string floor uint8 Key string } func (self Dept) Name() string { return self.name } func (self Dept) SetName(name string) { self.name = name } func (self *Dept) Relocate(building string, floor uint8) { self.building = building self.floor = floor }
而后我写了一些测试代码:
dept1 := Dept{ name: "MySohu", building: "Internet", floor: 7} switch v := interface{}(dept1).(type) { case DeptModeFull: fmt.Printf("The dept1 is a DeptModeFull.\n") case DeptModeB: fmt.Printf("The dept1 is a DeptModeB.\n") case DeptModeA: fmt.Printf("The dept1 is a DeptModeA.\n") default: fmt.Printf("The type of dept1 is %v\n", v) } deptPtr1 := &dept1 if _, ok := interface{}(deptPtr1).(DeptModeFull); ok { fmt.Printf("The deptPtr1 is a DeptModeFull.\n") } if _, ok := interface{}(deptPtr1).(DeptModeA); ok { fmt.Printf("The deptPtr1 is a DeptModeA.\n") } if _, ok := interface{}(deptPtr1).(DeptModeB); ok { fmt.Printf("The deptPtr1 is a DeptModeB.\n") }
打印出的内容:
The dept1 is a DeptModeA. The deptPtr1 is a DeptModeFull. The deptPtr1 is a DeptModeA. The deptPtr1 is a DeptModeB.
假设T是struct,那么Go里面遵循下面几个原则:
T的方法集仅拥有 T Receiver (方法中的接受者)方法。
*T 方法集则包含全部方法 (T + *T)。
所以你上面的例子dept1应该是拥有方法:Name和SetName
而&dept1拥有方法:Name、SetName和Relocate
这个就是Go里面在设计方法的时候需要注意Receiver的类型
Go语言中结构体方法副本传参与指针传参的区别
我们来看个例子:
package main import ( "fmt" ) type B struct { Name string } func(b B) Test1() { fmt.Printf("Test1 addr:%p\n", &b) fmt.Printf("Test1 name:%s\n", b.Name) b.Name = "john" } func(b *B) Test2() { fmt.Printf("Test2 addr:%p\n", b) fmt.Printf("Test2 name:%s\n", b.Name) b.Name = "john" } func main() { b := B{} b.Test1() b.Test1() b.Test2() b.Test2() }
执行后结果如下:
Test1 addr:0xc42000e1e0 Test1 name: Test1 addr:0xc42000e1f0 Test1 name: Test2 addr:0xc42000e1d0 Test2 name: Test2 addr:0xc42000e1d0 Test2 name:john
可以看到Test1中打印出b结构体的地址在变化,而Test2中没有变化,这说明每一次Test1的调用,都是传入的结构体b的一个副本(拷贝),当在Test1中对内部变量的任何改动,都将会失效(因为下一次访问的时候传入的是b结构体新的副本)。而Test2方法作为指针传参时,每一次传入的都是b结构体的指针,指向的是同一个结构体,因此地址没有变化,且对内部变量做改动时,都是改动的b结构体内容。
在Go语言中的这个差别可能是对OOP设计的一个坑,在Go语言中要想实现OOP的设计,在进行方法封装时,都采用Test2的写法。
感谢各位的阅读,以上就是“Go语言中结构体方法副本传参与指针传参的区别”的内容了,经过本文的学习后,相信大家对Go语言中结构体方法副本传参与指针传参的区别这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是创新互联成都网站设计公司,小编将为大家推送更多相关知识点的文章,欢迎关注!
另外有需要云服务器可以了解下创新互联scvps.cn,海内外云服务器15元起步,三天无理由+7*72小时售后在线,公司持有idc许可证,提供“云服务器、裸金属服务器、高防服务器、香港服务器、美国服务器、虚拟主机、免备案服务器”等云主机租用服务以及企业上云的综合解决方案,具有“安全稳定、简单易用、服务可用性高、性价比高”等特点与优势,专为企业上云打造定制,能够满足用户丰富、多元化的应用场景需求。