溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別

發(fā)布時(shí)間:2021-08-21 02:46:40 來(lái)源:億速云 閱讀:157 作者:chen 欄目:編程語(yǔ)言

這篇文章主要講解了“Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”吧!

GO語(yǔ)言結(jié)構(gòu)體方法跟結(jié)構(gòu)體指針?lè)椒ǖ膮^(qū)別

首先,我定了三個(gè)接口、一個(gè)結(jié)構(gòu)和三個(gè)方法:

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
}

而后我寫(xiě)了一些測(cè)試代碼:

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")
}

打印出的內(nèi)容:

The dept1 is a DeptModeA.?
The deptPtr1 is a DeptModeFull.
?The deptPtr1 is a DeptModeA.
?The deptPtr1 is a DeptModeB.

假設(shè)T是struct,那么Go里面遵循下面幾個(gè)原則:

  • T的方法集僅擁有 T Receiver (方法中的接受者)方法。

  • *T 方法集則包含全部方法 (T + *T)。

所以你上面的例子dept1應(yīng)該是擁有方法:Name和SetName

而&dept1擁有方法:Name、SetName和Relocate

這個(gè)就是Go里面在設(shè)計(jì)方法的時(shí)候需要注意Receiver的類(lèi)型

Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別

我們來(lái)看個(gè)例子:

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()
}

執(zhí)行后結(jié)果如下:

Test1 addr:0xc42000e1e0
Test1 name:
Test1 addr:0xc42000e1f0
Test1 name:
Test2 addr:0xc42000e1d0
Test2 name:
Test2 addr:0xc42000e1d0
Test2 name:john

可以看到Test1中打印出b結(jié)構(gòu)體的地址在變化,而Test2中沒(méi)有變化,這說(shuō)明每一次Test1的調(diào)用,都是傳入的結(jié)構(gòu)體b的一個(gè)副本(拷貝),當(dāng)在Test1中對(duì)內(nèi)部變量的任何改動(dòng),都將會(huì)失效(因?yàn)橄乱淮卧L問(wèn)的時(shí)候傳入的是b結(jié)構(gòu)體新的副本)。而Test2方法作為指針傳參時(shí),每一次傳入的都是b結(jié)構(gòu)體的指針,指向的是同一個(gè)結(jié)構(gòu)體,因此地址沒(méi)有變化,且對(duì)內(nèi)部變量做改動(dòng)時(shí),都是改動(dòng)的b結(jié)構(gòu)體內(nèi)容。

在Go語(yǔ)言中的這個(gè)差別可能是對(duì)OOP設(shè)計(jì)的一個(gè)坑,在Go語(yǔ)言中要想實(shí)現(xiàn)OOP的設(shè)計(jì),在進(jìn)行方法封裝時(shí),都采用Test2的寫(xiě)法。

感謝各位的閱讀,以上就是“Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Go語(yǔ)言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI