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
}
而后我寫了一些測(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的類型
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的寫法。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
GO語(yǔ)言類型轉(zhuǎn)換和類型斷言實(shí)例分析
這篇文章主要介紹了GO語(yǔ)言類型轉(zhuǎn)換和類型斷言,以實(shí)例形式詳細(xì)分析了類型轉(zhuǎn)換和類型斷言的概念與使用技巧,需要的朋友可以參考下2015-01-01
Golang實(shí)現(xiàn)Trie(前綴樹)的示例
本文主要介紹了Golang實(shí)現(xiàn)Trie(前綴樹)的示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-01-01
Go語(yǔ)言多人聊天室項(xiàng)目實(shí)戰(zhàn)
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言多人聊天室項(xiàng)目實(shí)戰(zhàn),實(shí)現(xiàn)單撩或多撩等多種功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-08-08
夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總
這篇文章主要8為大家介紹了夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總,有需要的朋友可以借鑒參考下,希望能夠有所幫助2023-10-10
gin正確多次讀取http?request?body內(nèi)容實(shí)現(xiàn)詳解
這篇文章主要為大家介紹了gin正確多次讀取http?request?body內(nèi)容實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01

