淺談Golang的方法傳遞值應(yīng)該注意的地方
其實(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ò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
golang使用net/rpc庫(kù)實(shí)現(xiàn)rpc
這篇文章主要為大家詳細(xì)介紹了golang如何使用net/rpc庫(kù)實(shí)現(xiàn)rpc,文章的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,需要的小伙伴可以參考一下2024-01-01基于gin的golang web開(kāi)發(fā):路由示例詳解
這篇文章主要介紹了基于gin的golang web開(kāi)發(fā):路由示例詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10golang實(shí)現(xiàn)簡(jiǎn)單的udp協(xié)議服務(wù)端與客戶端示例
這篇文章主要介紹了golang實(shí)現(xiàn)簡(jiǎn)單的udp協(xié)議服務(wù)端與客戶端,結(jié)合實(shí)例形式分析了基于UDP協(xié)議的數(shù)據(jù)傳輸相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2016-07-07Golang信號(hào)處理及如何實(shí)現(xiàn)進(jìn)程的優(yōu)雅退出詳解
這篇文章主要給大家介紹了關(guān)于Golang信號(hào)處理及如何實(shí)現(xiàn)進(jìn)程的優(yōu)雅退出的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2018-03-03詳解Golang如何實(shí)現(xiàn)支持隨機(jī)刪除元素的堆
堆是一種非常常用的數(shù)據(jù)結(jié)構(gòu),它能夠支持在O(1)的時(shí)間復(fù)雜度獲取到最大值(或最小值)。本文主要介紹了如何實(shí)現(xiàn)支持O(log(n))隨機(jī)刪除元素的堆,需要的可以參考一下2022-09-09go語(yǔ)言中fallthrough的用法說(shuō)明
這篇文章主要介紹了go語(yǔ)言中fallthrough的用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05Golang中實(shí)現(xiàn)數(shù)據(jù)脫敏處理的go-mask包分享
這篇文章主要是來(lái)和大家分享一個(gè)在輸出中對(duì)敏感數(shù)據(jù)進(jìn)行脫敏的工作包:go-mask,可以將敏感信息輸出的時(shí)候替換成星號(hào)或其他字符,感興趣的小編可以跟隨小編一起了解下2023-05-05