前段時間,我們實驗室用go作為后臺開發(fā)語言開發(fā)了一個web項目,由于這是自己第一次使用go語言進(jìn)行開發(fā),在開發(fā)過程中,一味著追求完成任務(wù),在編碼的時候沒有太注重性能,雖然勉強實現(xiàn)了功能,但是對go語言的理解還是比較淺顯的。下面來談?wù)勛约簩o語言中函數(shù)與方法的理解。
成都創(chuàng)新互聯(lián)是網(wǎng)站建設(shè)專家,致力于互聯(lián)網(wǎng)品牌建設(shè)與網(wǎng)絡(luò)營銷,專業(yè)領(lǐng)域包括成都網(wǎng)站設(shè)計、成都網(wǎng)站建設(shè)、電商網(wǎng)站制作開發(fā)、微信小程序開發(fā)、微信營銷、系統(tǒng)平臺開發(fā),與其他網(wǎng)站設(shè)計及系統(tǒng)開發(fā)公司不同,我們的整合解決方案結(jié)合了恒基網(wǎng)絡(luò)品牌建設(shè)經(jīng)驗和互聯(lián)網(wǎng)整合營銷的理念,并將策略和執(zhí)行緊密結(jié)合,且不斷評估并優(yōu)化我們的方案,為客戶提供全方位的互聯(lián)網(wǎng)品牌整合方案!
普通函數(shù):
go函數(shù)可以返回多個值
值傳遞: 值傳遞是指在調(diào)用函數(shù)時將實際參數(shù)復(fù)制一份傳遞到函數(shù)中,這樣函數(shù)中如果對參數(shù)進(jìn)行修改,將不會影響到實際參數(shù)
引用傳遞: 引用傳遞是指在調(diào)用函數(shù)將實際參數(shù)的地址傳遞到函數(shù)中,那么在函數(shù)中對參數(shù)進(jìn)行的修改,將影響到實際參數(shù)。
一般來說go語言函數(shù)的 接收者(也就是形參)一般放在函數(shù)名后面 ,不能將指針類型的數(shù)據(jù)直接傳遞,也就是說函數(shù)形參如果是值類型,調(diào)用者必須使用值作為實參過來,如果函數(shù)形參是指針類型,則函數(shù)調(diào)用者需使用指針作為實參來調(diào)用。
普通方法:
接收者是在func關(guān)鍵字后面,而不是在函數(shù)名稱后面,接收者可以是自己定義的一個類型,這個類型可以是struct、interface,一個方法就是一個包含了接收者的函數(shù),接收者可以是命名類型或者是結(jié)構(gòu)體類型的一個值或者是一個指針。
下面是一個例子來說明方法和函數(shù)的區(qū)別(重點)
1、值接收者和指針接收者
所謂指針接收者和值接收者這兩個概念,用GO寫了一陣子代碼的人都了解了,這里只做簡要說明一下,也就是對于一個給定結(jié)構(gòu),咱們對結(jié)構(gòu)進(jìn)行方法包裝的時候,固定必傳的參數(shù),用來指向這個對象結(jié)構(gòu)自身的一個參數(shù),在go中也就是形式如下:
我們對結(jié)構(gòu)體testStruct進(jìn)行了包裝,提供了兩個方法,sum和modify,其中sum的方法接收者為a testStruct,這個就是值接收者,而modify的接收者為a *testStruct就是指針接收者,也就是說固定對象指針,一個傳遞的是指針地址,而另外一個直接傳遞的是結(jié)構(gòu)值拷貝了
對指針有一定了解的,都可以知道,指針傳遞過去的,可以直接修改結(jié)構(gòu)內(nèi)部內(nèi)容,而值傳遞過去的,無論如何修改這個接收者的數(shù)據(jù),不會對原對象結(jié)構(gòu)產(chǎn)生影響。而對于咱們包裝結(jié)構(gòu)對象的時候,到底是使用指針還是使用值接收者,這個實際上沒有太大的定論,就我個人的觀點來說,如果結(jié)構(gòu)體占有的內(nèi)存空間不大(kb級別),而又不需要修改內(nèi)部的,同時結(jié)構(gòu)對象內(nèi)部沒有同步對象比如(sync包中的mutex,rwlock,waitgroup等之類的結(jié)構(gòu)的話,可以直接值傳遞,實際上值copy也沒有咱們想象的那么慢,很多時候,都用指針,最后的gc回收掃描可能都比咱們這個傳遞copy的消耗大) p="" /kb級別),而又不需要修改內(nèi)部的,同時結(jié)構(gòu)對象內(nèi)部沒有同步對象比如(sync包中的mutex,rwlock,waitgroup等之類的結(jié)構(gòu)的話,可以直接值傳遞,實際上值copy也沒有咱們想象的那么慢,很多時候,都用指針,最后的gc回收掃描可能都比咱們這個傳遞copy的消耗大)
2、實現(xiàn)接口的值接收者和指針接收者有啥區(qū)別
也就是比如定義如下
這里面的值接收者和指針接收者有什么區(qū)別,這里咱來寫一個測試
通過這個測試用例可以發(fā)現(xiàn),指針接收者實現(xiàn)的接口可以同時支持轉(zhuǎn)移到值接收者接口和指針接收者接口,而用值接收者實現(xiàn)的接口,則無法轉(zhuǎn)移到使用指針接收者實現(xiàn)的接口,為啥子呢?目前網(wǎng)上或者各類資料上都是給的一個很官方很官方,而且很書面話難以理解的說明,大致意思如下:
這是目前網(wǎng)絡(luò)或者各種資料上都是差不多是這樣說的,看似講了,實際上就說了一個結(jié)果,根本就沒說出來一個為什么。這樣的總結(jié)出來,一個初學(xué)者的角度來看,是很不好理解的,初學(xué)者要么就是死記硬背,要么就是生搬硬套,甚至直到寫了好多好多代碼了,都還沒有搞明白一個為啥子,只是會用了而已,從長遠(yuǎn)來說這是不利于自身提高的。
有這兩個本質(zhì)點,咱們自己來思考一下,如果你來實現(xiàn)這個編譯器的時候,用指針接收的時候,指針接收者,默認(rèn)就能直接獲取支持,而值接收者實現(xiàn)接口的咱們可以直接來一個解指針就變成了值,就能匹配上值接收者實現(xiàn)的接口了,反過來說,如果值接收者,此時要匹配指針接收者,如何匹配呢,取一個地址就變成了指針了,此時數(shù)據(jù)類型確實是匹配了,但是,地址指向的數(shù)據(jù)區(qū)不對了,因為我們剛剛說了值接收者拷貝了一個新值之后是完全的一個新的對象,這個新對象和原始對象一點關(guān)系都沒有,咱們?nèi)〉刂?,取的也是這個新對象地址,對這個地址進(jìn)行操作,也是這個新對象的內(nèi)部數(shù)據(jù),和原始數(shù)據(jù)內(nèi)部沒有任何關(guān)系,所以由此就能推斷出,這個是為啥子值接收者不能匹配上指針接收者,而指針接收者卻可以匹配上值接收者了。
1、在某個作用域內(nèi)部,所有定義的字符串的數(shù)據(jù)區(qū)相同
這個很好驗證,代碼如下:
2、字符串相加會產(chǎn)生一個新串
這個也很好驗證
3、字符串真的是不可變的嗎
實際上從字符串的結(jié)構(gòu)
從這個結(jié)構(gòu),就能大致的推斷出來,字符串設(shè)計成這樣就不具備直接擴容+來增加新數(shù)據(jù),而如果咱們直接使用string[index] = 'a',用這種方式,就不能編譯通過,官方也確定說字符串是不可變的。那么真的是不可變的嗎?
通過上面的結(jié)構(gòu),在加上go的slice切片的數(shù)據(jù)結(jié)構(gòu)
由此可見,咱們可以將字符串通過指針方式強轉(zhuǎn)為一個byte數(shù)組指針,然后通過byte切片來修改,試試
編譯通過,運行報錯
unexpected fault address 0xae2e27
fatal error: fault
這個錯誤,基本上就是一個內(nèi)存的保護(hù)錯誤,是寫異常,所以說明了,這個肯定做了內(nèi)存寫保護(hù),那么直接修改一下內(nèi)存區(qū)的屬性,去掉他的寫保護(hù),就能寫了
以下代碼都是在Win平臺,Go1.18,Win上修改內(nèi)存權(quán)限屬性,使用VirtualProtect,代碼如下
此時運行,就能發(fā)現(xiàn)tstr的內(nèi)容被咱們變了,這種情況實際上在實際開發(fā)中不具有實際意義,因為本身在語言層面,已經(jīng)做了層層限制,咱們這是屬于非法強制的操作方式,是流氓行為,那么是否有比較溫和一點的操作方式呢?答案是有的,且往下看。
通過上面,我們已經(jīng)用到了字符串結(jié)構(gòu),切片結(jié)構(gòu),要想字符串內(nèi)容可變,那么咱們自己構(gòu)造字符串的數(shù)據(jù)內(nèi)容區(qū)域,且讓這個數(shù)據(jù)區(qū)木有內(nèi)存寫保護(hù)不就行了,內(nèi)容區(qū)可變,GO原生態(tài)的byte數(shù)組不就行嘛,所以咱們自己構(gòu)造一下
此時我們直接修改buffer的內(nèi)容,就是直接修改了str的數(shù)據(jù)內(nèi)容了。而又不會像前面的一樣遇到內(nèi)存寫保護(hù)
4、字符串轉(zhuǎn)換優(yōu)化時可能碰到的坑
通過前面討論的字符串的可變性的方法,咱們可以知道,很多時候,[]byte到字符串的轉(zhuǎn)變,可以直接構(gòu)造其結(jié)構(gòu),而共享數(shù)據(jù),從而達(dá)到減少數(shù)據(jù)內(nèi)存copy的方式來進(jìn)行優(yōu)化,再使用這些優(yōu)化的時候,一定需要注意,字符串或者數(shù)組的生命周期,是否會存在被改寫的情況,從而導(dǎo)致前后不一致的問題。
比如下面這段代碼:
大家可以猜想一下,這個最后里面的數(shù)據(jù)mmp中,"test"的value是多少,"abcd"的value是多少,然后想想為什么,且等端午之后,再來分解
go語言沒有面向?qū)ο蟮奶匦?,也沒有類對象的概念。但是,可以使用結(jié)構(gòu)體來模擬這些特性,我們都知道面向?qū)ο罄锩嬗蓄惙椒ǖ雀拍睢N覀円部梢月暶饕恍┓椒?,屬于某個結(jié)構(gòu)體。
Go中的方法,是一種特殊的函數(shù),定義域struct之上(與struct關(guān)聯(lián)、綁定),被稱為struct的接受者(receiver)。通俗的講,方法就是有接收者的函數(shù)。
語法格式如下:
mytype:定義一個結(jié)構(gòu)體
recv:接受該方法的結(jié)構(gòu)體(receiver)
my_method:方法名稱
para:參數(shù)列表
return_type:返回值類型
從語法格式可以看出,一個方法和一個函數(shù)非常相似,多了一個接受類型。
實例
運行結(jié)果
目錄
一、結(jié)構(gòu)體詳解
1. 結(jié)構(gòu)體定義
2. 實例化結(jié)構(gòu)體的7種方法
二、結(jié)構(gòu)體方法
1. 結(jié)構(gòu)體的方法定義
2. 結(jié)構(gòu)體內(nèi)自定義方法的引用
3. 任意類型添加方法
三、嵌套、繼承
1. 匿名結(jié)構(gòu)體
2. 結(jié)構(gòu)體中可以定義任意類型的字段
3. 結(jié)構(gòu)體嵌套結(jié)構(gòu)體
4. 結(jié)構(gòu)體嵌套匿名結(jié)構(gòu)體
5. 結(jié)構(gòu)體嵌套多個匿名結(jié)構(gòu)體
6. 結(jié)構(gòu)體繼承
四、結(jié)構(gòu)體和JSON相互轉(zhuǎn)換
1. 結(jié)構(gòu)體轉(zhuǎn)化成json
2. json轉(zhuǎn)化成結(jié)構(gòu)體
3. 結(jié)構(gòu)體標(biāo)簽 tag
4. 嵌套結(jié)構(gòu)體和json的序列化反序列化
Golang 中沒有“類”的概念,Golang 中的結(jié)構(gòu)體和其他語言中的類有點相似。和其他面向?qū)?象語言中的類相比,Golang 中的結(jié)構(gòu)體具有更高的擴展性和靈活性。
Golang 中的基礎(chǔ)數(shù)據(jù)類型可以表示一些事物的基本屬性,但是當(dāng)我們想表達(dá)一個事物的全 部或部分屬性時,這時候再用單一的基本數(shù)據(jù)類型就無法滿足需求了,Golang 提供了一種 自定義數(shù)據(jù)類型,可以封裝多個基本數(shù)據(jù)類型,這種數(shù)據(jù)類型叫結(jié)構(gòu)體,英文名稱 struct。 也就是我們可以通過 struct 來定義自己的類型了。
使用 type 和 struct 關(guān)鍵字來定義結(jié)構(gòu)體,具體代碼格式如下:
type 類型名 struct {
字段名 字段類型
字段名 字段類型 …
}
其中:
? 類型名:表示自定義結(jié)構(gòu)體的名稱,在同一個包內(nèi)不能重復(fù)。
? 字段名:表示結(jié)構(gòu)體字段名。結(jié)構(gòu)體中的字段名必須唯一。
? 字段類型:表示結(jié)構(gòu)體字段的具體類型。
在 go 語言中,沒有類的概念但是可以給類型(結(jié)構(gòu)體,自定義類型)定義方法。所謂方法 就是定義了接收者的函數(shù)。接收者的概念就類似于其他語言中的 this 或者 self。
方法的定義格式如下:
func (接收者變量 接收者類型) 方法名(參數(shù)列表) (返回參數(shù)) {
函數(shù)體
}
注意:想改變結(jié)構(gòu)體內(nèi)的值,必須先變成指針。
在 Go 語言中,接收者的類型可以是任何類型,不僅僅是結(jié)構(gòu)體,任何類型都可以擁有方法。 舉個例子,我們基于內(nèi)置的 int 類型使用 type 關(guān)鍵字可以定義新的自定義類型,然后為我們 的自定義類型添加方法。
注意:匿名結(jié)構(gòu)體中不允許出現(xiàn)多個重復(fù)的類型
注意:如果結(jié)構(gòu)體里面有私有屬性也就是小寫定義的字段,則不會被json使用
因為結(jié)構(gòu)Student和Teacher實現(xiàn)接口Human的方法SayHello時,接受的是通過一個指針類型的變量(見(s *Student)和(t *Teacher))來調(diào)用這個方法。因此,在調(diào)用SayHi函數(shù)時,只能傳遞Student或Teacher的對象的地址,傳遞它們的對象是錯的。
相反,如果結(jié)構(gòu)Student和Teacher實現(xiàn)接口Human的方法SayHello時,接受的是通過一個對象(像(s Student)和(t Teacher))來調(diào)用這個方法。則在調(diào)用SayHi函數(shù)時,既能傳遞Student或Teacher的對象,也能傳遞Student或Teacher的對象的地址。
goget請求可以接受結(jié)構(gòu)體。
接收者是結(jié)構(gòu)體時,可以是結(jié)構(gòu)體類型、結(jié)構(gòu)體指針類型。調(diào)用時不區(qū)分調(diào)用者是結(jié)構(gòu)體還是結(jié)構(gòu)體指針,go語言會自動轉(zhuǎn)化為對應(yīng)的結(jié)構(gòu)體或結(jié)構(gòu)體指針。