91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Go語言中結構體方法副本傳參與指針傳參的區別

發布時間:2021-08-21 02:46:40 來源:億速云 閱讀:165 作者:chen 欄目:編程語言

這篇文章主要講解了“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語言中結構體方法副本傳參與指針傳參的區別這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

平舆县| 连山| 南投市| 杭州市| 大关县| 桂东县| 西宁市| 高台县| 碌曲县| 平顺县| 九龙城区| 三门县| 板桥市| 资兴市| 当阳市| 新巴尔虎右旗| 乡宁县| 钟祥市| 特克斯县| 娄底市| 河津市| 汉源县| 普洱| 平定县| 锡林郭勒盟| 南岸区| 遵义县| 沾化县| 宁乡县| 遂平县| 枣庄市| 滁州市| 六安市| 额敏县| 渭源县| 布拖县| 叙永县| 吴江市| 桓仁| 那坡县| 凤城市|