其實(shí)最近看了不少Golang接口以及方法的闡述都有一個(gè)地方?jīng)]說(shuō)得特別明白。就是在Golang編譯隱式轉(zhuǎn)換傳遞給方法使用的時(shí)候,和調(diào)用函數(shù)時(shí)的區(qū)別。
我們都知道,在我們?yōu)橐粋€(gè)類(lèi)型變量申明了一個(gè)方法的時(shí)候,我們可以使用類(lèi)似于self.method來(lái)調(diào)用這個(gè)方法,而且無(wú)論你申明的方法的接收器是指針接收器還是值接收器,Golang都可以幫你隱式轉(zhuǎn)換為正確的值供方法使用。
讓我們來(lái)看一個(gè)例子:
package main
import "fmt"
type duration int
func (d *duration) pretty() string {
return fmt.Sprintf("Duration: %d", d)
}
func main() {
var kk duration
kk = 3
kk.pretty()
}
在這個(gè)例子中,創(chuàng)建了一個(gè)類(lèi)型為duration的變量kk,并且duration這個(gè)類(lèi)型上有指針接收器的方法pretty()這個(gè)時(shí)候無(wú)論你使用kk.pretty()還有使用(kk).pretty()都會(huì)正確執(zhí)行,并且就算接收器不是指針類(lèi)型而是值類(lèi)型,同上一樣。Golang編譯器會(huì)將你傳入的值隱式轉(zhuǎn)換為正確的傳入對(duì)象。
這個(gè)不難理解,但是有一個(gè)跟他很像的特性,卻會(huì)讓這個(gè)問(wèn)題變得很繞。那就是調(diào)用接口的時(shí)候出現(xiàn)的情況
同樣我們來(lái)看一個(gè)例子:
package main
import (
"fmt"
)
type notifier interface {
notify()
}
type user struct {
name string
email string
}
func(u *user) notify() {
fmt.Printf("Sending user email to %s%s>\n",
u.name,
u.email)
}
func sendNotification(n notifier) {
n.notify()
}
func main() {
u := user{"Bill", "bill@xiachufang.com"}
sendNotification(u)
}
這個(gè)例子就不是用類(lèi)型直接調(diào)用自己的方法了,而是把自己當(dāng)作參數(shù)傳遞給接口。讓接口去執(zhí)行對(duì)應(yīng)方法。
這里注意,接口對(duì)于類(lèi)型的要求就十分嚴(yán)格了,接口在神明的時(shí)候會(huì)指定,擁有哪些方法(這里的方法指 方法名, 方法參數(shù),以及方法返回類(lèi)型)。實(shí)現(xiàn)了這些方法就實(shí)現(xiàn)了這個(gè)接口。這里我們調(diào)用sendNotification這個(gè)方法需要傳遞進(jìn)實(shí)現(xiàn)了notifier這個(gè)接口的變量做參數(shù)。查看notifier代碼可以注意到,他實(shí)現(xiàn)了一個(gè)notify的方法。而我們的user實(shí)現(xiàn)了一個(gè)指針接收器的notify方法。但是這里注意,傳遞值必須遵守一個(gè)條件即:
如果接口實(shí)現(xiàn)方法,類(lèi)型自己的實(shí)現(xiàn)使用的是值接收器,那么在傳遞值的時(shí)候無(wú)論使用指針還是值都可以。
如果接口實(shí)現(xiàn)方法,類(lèi)型自己的實(shí)現(xiàn)使用的是指針接收器,那么在傳遞值的時(shí)候必須傳遞地址。
所以上面的例子,接收器是指針接收器,我們必須傳遞地址,如果傳遞值則會(huì)報(bào)錯(cuò)。
那么是為什么這里又不能進(jìn)行隱式轉(zhuǎn)換了呢?
實(shí)際上是因?yàn)?,編譯器并不能總能自動(dòng)獲得一個(gè)值的地方,也就是說(shuō)你傳u,編譯器不一定能知道u的地址是啥。。他可能沒(méi)有辦法幫你完成轉(zhuǎn)換。
補(bǔ)充:Golang 數(shù)組(切片)的值傳遞與引用傳遞
Go語(yǔ)言中函數(shù)的參數(shù)都是按值進(jìn)行傳遞的,即使參數(shù)是指針,也是指針的一個(gè)副本。習(xí)慣上把指針的函數(shù)參數(shù)稱(chēng)之為地址傳參,即引用傳遞,而非指針的函數(shù)參數(shù)稱(chēng)為值傳參
地址傳參在大對(duì)象上效率比值傳參好,在內(nèi)部相當(dāng)于用指針地址賦值,而不用復(fù)制整個(gè)對(duì)象
一、數(shù)組的值傳遞
Golang數(shù)組作為參數(shù)傳入函數(shù)時(shí),進(jìn)行的是值傳遞,這里與Java數(shù)組的引用傳遞是不同的,示例如下
package main
import "fmt"
func main() {
arr := [8]int{}
for i := 0; i 8; i++ {
arr[i] = i
}
fmt.Println(arr)
exchange(arr)
fmt.Println(arr)
}
func exchange(arr [8]int) {
for k, v := range arr {
arr[k] = v * 2
}
}
運(yùn)行結(jié)果如下:

二、數(shù)組的引用傳遞
默認(rèn)情況下Golang的數(shù)組傳遞是值傳遞,但當(dāng)我們想要對(duì)傳入的數(shù)組進(jìn)行修改時(shí),可以使用指針來(lái)對(duì)數(shù)組進(jìn)行操作,如下
package main
import "fmt"
func main() {
arr := [8]int{}
for i := 0; i 8; i++ {
arr[i] = i
}
fmt.Println(arr)
exchangeByAddress(arr)
fmt.Println(arr)
}
func exchangeByAddress(arr *[8]int) {
for k, v := range *arr {
arr[k] = v * 2
}
}
運(yùn)行結(jié)果如下:

三、切片的引用傳遞
Golang中的切片與Java中的ArrayList集合類(lèi)似,進(jìn)行的是引用傳遞
package main
import "fmt"
func main() {
slice := []int{1,2,3,4,5}
fmt.Println(slice)
exchangeSlice(slice)
fmt.Println(slice)
}
func exchangeSlice(slice []int) {
for k, v := range slice {
slice[k] = v * 2
}
}
運(yùn)行結(jié)果:

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- Golang數(shù)組的傳遞詳解
- Golang中的參數(shù)傳遞示例詳解
- Golang的md5 hash計(jì)算操作
- golang gin 框架 異步同步 goroutine 并發(fā)操作
- golang數(shù)組-----尋找數(shù)組中缺失的整數(shù)方法