亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

Dept的實例的指針被判定為全部三個接口的實現,而Dept的實例只是DeptModeA接口的實現?

Dept的實例的指針被判定為全部三個接口的實現,而Dept的實例只是DeptModeA接口的實現?

Go
千巷貓影 2023-05-02 11:11:55
直接上代碼:首先,我定了三個接口、一個結構和三個方法:type DeptModeFull interface {    Name() string     SetName(name string)     Relocate(building string, floor uint8)}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 := &dept1if _, 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.哪位大佬給講一下?
查看完整描述

3 回答

?
藍山帝景

TA貢獻1843條經驗 獲得超7個贊

假設T是struct,那么Go里面遵循下面幾個原則:

  • T的方法集僅擁有 T Receiver 方法。

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

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

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

這個就是Go里面在設計方法的時候需要注意Receiver的類型

查看完整回答
反對 回復 2023-05-05
?
RISEBY

TA貢獻1856條經驗 獲得超5個贊

1、結構Dept的方法集中僅包含方法接收者為Dept的方法,即:Name()SetName())。所以,結構Dept的實例僅為DeptModeA的實現。
2、結構的指針*Dept的方法集包含了方法接受者為Dept*Dept的方法,即:Name()、SetName()Relocate()。所以,接口Dept的實例的指針為全部三個接口——DeptModeFull、DeptModeADeptModeB的實現。 

但是我覺得上述設計和原則與方法調用的設計有些不一致,請看下面的代碼:

dept1.Relocate("Media", 12)fmt.Printf("Dept: %v\n", dept1)fmt.Printf("Dept name: %v\n", deptPtr1.Name())

打印的內容:

Dept: {MySohu Media 12 }?Dept name: MySohu

這說明:結構Dept的實例可以調用其指針方法集中的方法。

我覺得設計不一致的地方是:既然結構的訪法集不包含接收者為其指針的方法,那又為什么讓在結構的實例上可以調用其指針訪法集中的方法呢?反過來看,既然結構的實例可以調用其接口訪法集上的方法,那為什么不讓結構的方法集直接包含這些方法呢?

不過,不管怎樣,這引出了第三條規則(官方規范的意譯):
3、如果結構的實例x是“可被尋址的”,且&x的方法集中包含方法m,則x.m()(&x).m()的速記(快捷方式)。

查看完整回答
反對 回復 2023-05-05
?
偶然的你

TA貢獻1841條經驗 獲得超3個贊

所以其實

dept1.Relocate("Media", 12)

被自動轉化成了?

(&dept1).Relocate("Media", 12)

學習了。

可尋址的意思大概就是已經實例話了。如果你直接

var deptPtr2 *Dept
deptPtr2.Relocate("Media", 12)

就不行。反之:

var deptPtr2 Dept
deptPtr2.Relocate("Media", 12)

就可以。區別就在這里了。

查看完整回答
反對 回復 2023-05-05
  • 3 回答
  • 0 關注
  • 183 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號