這篇文章主要講解了“Go語言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Go語言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”吧!
為湯旺等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計(jì)制作服務(wù),及湯旺網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為網(wǎng)站建設(shè)、網(wǎng)站制作、湯旺網(wǎng)站設(shè)計(jì),以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會得到認(rèn)可,從而選擇與我們長期合作。這樣,我們也可以走得更遠(yuǎn)!GO語言結(jié)構(gòu)體方法跟結(jié)構(gòu)體指針方法的區(qū)別
首先,我定了三個接口、一個結(jié)構(gòu)和三個方法:
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") }
打印出的內(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里面遵循下面幾個原則:
T的方法集僅擁有 T Receiver (方法中的接受者)方法。
*T 方法集則包含全部方法 (T + *T)。
所以你上面的例子dept1應(yīng)該是擁有方法:Name和SetName
而&dept1擁有方法:Name、SetName和Relocate
這個就是Go里面在設(shè)計(jì)方法的時候需要注意Receiver的類型
Go語言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別
我們來看個例子:
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中沒有變化,這說明每一次Test1的調(diào)用,都是傳入的結(jié)構(gòu)體b的一個副本(拷貝),當(dāng)在Test1中對內(nèi)部變量的任何改動,都將會失效(因?yàn)橄乱淮卧L問的時候傳入的是b結(jié)構(gòu)體新的副本)。而Test2方法作為指針傳參時,每一次傳入的都是b結(jié)構(gòu)體的指針,指向的是同一個結(jié)構(gòu)體,因此地址沒有變化,且對內(nèi)部變量做改動時,都是改動的b結(jié)構(gòu)體內(nèi)容。
在Go語言中的這個差別可能是對OOP設(shè)計(jì)的一個坑,在Go語言中要想實(shí)現(xiàn)OOP的設(shè)計(jì),在進(jìn)行方法封裝時,都采用Test2的寫法。
感謝各位的閱讀,以上就是“Go語言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Go語言中結(jié)構(gòu)體方法副本傳參與指針傳參的區(qū)別這一問題有了更深刻的體會,具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián)成都網(wǎng)站設(shè)計(jì)公司,小編將為大家推送更多相關(guān)知識點(diǎn)的文章,歡迎關(guān)注!
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點(diǎn)與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。