import "workname/packetfolder"
成都創(chuàng)新互聯(lián)公司秉承實(shí)現(xiàn)全網(wǎng)價(jià)值營(yíng)銷(xiāo)的理念,以專(zhuān)業(yè)定制企業(yè)官網(wǎng),做網(wǎng)站、網(wǎng)站制作,微信小程序,網(wǎng)頁(yè)設(shè)計(jì)制作,手機(jī)網(wǎng)站開(kāi)發(fā),成都全網(wǎng)營(yíng)銷(xiāo)幫助傳統(tǒng)企業(yè)實(shí)現(xiàn)“互聯(lián)網(wǎng)+”轉(zhuǎn)型升級(jí)專(zhuān)業(yè)定制企業(yè)官網(wǎng),公司注重人才、技術(shù)和管理,匯聚了一批優(yōu)秀的互聯(lián)網(wǎng)技術(shù)人才,對(duì)客戶(hù)都以感恩的心態(tài)奉獻(xiàn)自己的專(zhuān)業(yè)和所長(zhǎng)。
導(dǎo)入多個(gè)包
方法調(diào)用 包名.函數(shù)//不是函數(shù)或結(jié)構(gòu)體所處文件或文件夾名
packagename.Func()
前面加個(gè)點(diǎn)表示省略調(diào)用,那么調(diào)用該模塊里面的函數(shù),可以不用寫(xiě)模塊名稱(chēng)了:
當(dāng)導(dǎo)入一個(gè)包時(shí),該包下的文件里所有init()函數(shù)都會(huì)被執(zhí)行,然而,有些時(shí)候我們并不需要把整個(gè)包都導(dǎo)入進(jìn)來(lái),僅僅是是希望它執(zhí)行init()函數(shù)而已。下劃線(xiàn)的作用僅僅是為了調(diào)用init()函數(shù),所以無(wú)法通過(guò)包名來(lái)調(diào)用包中的其他函數(shù)
import _ package
變量聲明必須要使用否則會(huì)報(bào)錯(cuò)。
全局變量運(yùn)行聲明但不使用。
func 函數(shù)名 (參數(shù)1,參數(shù)2,...) (返回值a 類(lèi)型a, 返回值b 類(lèi)型b,...)
func 函數(shù)名 (參數(shù)1,參數(shù)2,...) (返回值類(lèi)型1, 返回值類(lèi)型2,...)
func (this *結(jié)構(gòu)體名) 函數(shù)名(參數(shù) string) (返回值類(lèi)型1, 返回值類(lèi)型2){}
使用大小來(lái)區(qū)分函數(shù)可見(jiàn)性
大寫(xiě)是public類(lèi)型
小寫(xiě)是private類(lèi)型
func prifunc int{}
func pubfunc int{}
聲明靜態(tài)變量
const value int
定義變量
var value int
聲明一般類(lèi)型、接口和結(jié)構(gòu)體
聲明函數(shù)
func function () int{}
go里面所有的空值對(duì)應(yīng)如下
通道類(lèi)型
內(nèi)建函數(shù) new 用來(lái)分配內(nèi)存,它的第一個(gè)參數(shù)是一個(gè)類(lèi)型,不是一個(gè)值,它的返回值是一個(gè)指向新分配類(lèi)型零值的指針
func new(Type) *Type
[這位博主有非常詳細(xì)的分析]
Go 語(yǔ)言支持并發(fā),我們只需要通過(guò) go 關(guān)鍵字來(lái)開(kāi)啟 goroutine 即可。
goroutine 是輕量級(jí)線(xiàn)程,goroutine 的調(diào)度是由 Golang 運(yùn)行時(shí)進(jìn)行管理的。
同一個(gè)程序中的所有 goroutine 共享同一個(gè)地址空間。
語(yǔ)法格式如下:
通道(channel)是用來(lái)傳遞數(shù)據(jù)的一個(gè)數(shù)據(jù)結(jié)構(gòu)。
通道的聲明
通道可用于兩個(gè) goroutine 之間通過(guò)傳遞一個(gè)指定類(lèi)型的值來(lái)同步運(yùn)行和通訊。操作符 - 用于指定通道的方向,發(fā)送或接收。如果未指定方向,則為雙向通道。
[這里有比較詳細(xì)的用例]
go里面的空接口可以指代任何類(lèi)型(無(wú)論是變量還是函數(shù))
聲明空接口
go里面的的強(qiáng)制類(lèi)型轉(zhuǎn)換語(yǔ)法為:
int(data)
如果是接口類(lèi)型的強(qiáng)制轉(zhuǎn)成其他類(lèi)型的語(yǔ)法為:
go里面的強(qiáng)制轉(zhuǎn)換是將值復(fù)制過(guò)去,所以在數(shù)據(jù)量的時(shí)候有比較高的運(yùn)行代價(jià)
目錄
一、結(jié)構(gòu)體詳解
1. 結(jié)構(gòu)體定義
2. 實(shí)例化結(jié)構(gòu)體的7種方法
二、結(jié)構(gòu)體方法
1. 結(jié)構(gòu)體的方法定義
2. 結(jié)構(gòu)體內(nèi)自定義方法的引用
3. 任意類(lèi)型添加方法
三、嵌套、繼承
1. 匿名結(jié)構(gòu)體
2. 結(jié)構(gòu)體中可以定義任意類(lèi)型的字段
3. 結(jié)構(gòu)體嵌套結(jié)構(gòu)體
4. 結(jié)構(gòu)體嵌套匿名結(jié)構(gòu)體
5. 結(jié)構(gòu)體嵌套多個(gè)匿名結(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 中沒(méi)有“類(lèi)”的概念,Golang 中的結(jié)構(gòu)體和其他語(yǔ)言中的類(lèi)有點(diǎn)相似。和其他面向?qū)?象語(yǔ)言中的類(lèi)相比,Golang 中的結(jié)構(gòu)體具有更高的擴(kuò)展性和靈活性。
Golang 中的基礎(chǔ)數(shù)據(jù)類(lèi)型可以表示一些事物的基本屬性,但是當(dāng)我們想表達(dá)一個(gè)事物的全 部或部分屬性時(shí),這時(shí)候再用單一的基本數(shù)據(jù)類(lèi)型就無(wú)法滿(mǎn)足需求了,Golang 提供了一種 自定義數(shù)據(jù)類(lèi)型,可以封裝多個(gè)基本數(shù)據(jù)類(lèi)型,這種數(shù)據(jù)類(lèi)型叫結(jié)構(gòu)體,英文名稱(chēng) struct。 也就是我們可以通過(guò) struct 來(lái)定義自己的類(lèi)型了。
使用 type 和 struct 關(guān)鍵字來(lái)定義結(jié)構(gòu)體,具體代碼格式如下:
type 類(lèi)型名 struct {
字段名 字段類(lèi)型
字段名 字段類(lèi)型 …
}
其中:
? 類(lèi)型名:表示自定義結(jié)構(gòu)體的名稱(chēng),在同一個(gè)包內(nèi)不能重復(fù)。
? 字段名:表示結(jié)構(gòu)體字段名。結(jié)構(gòu)體中的字段名必須唯一。
? 字段類(lèi)型:表示結(jié)構(gòu)體字段的具體類(lèi)型。
在 go 語(yǔ)言中,沒(méi)有類(lèi)的概念但是可以給類(lèi)型(結(jié)構(gòu)體,自定義類(lèi)型)定義方法。所謂方法 就是定義了接收者的函數(shù)。接收者的概念就類(lèi)似于其他語(yǔ)言中的 this 或者 self。
方法的定義格式如下:
func (接收者變量 接收者類(lèi)型) 方法名(參數(shù)列表) (返回參數(shù)) {
函數(shù)體
}
注意:想改變結(jié)構(gòu)體內(nèi)的值,必須先變成指針。
在 Go 語(yǔ)言中,接收者的類(lèi)型可以是任何類(lèi)型,不僅僅是結(jié)構(gòu)體,任何類(lèi)型都可以擁有方法。 舉個(gè)例子,我們基于內(nèi)置的 int 類(lèi)型使用 type 關(guān)鍵字可以定義新的自定義類(lèi)型,然后為我們 的自定義類(lèi)型添加方法。
注意:匿名結(jié)構(gòu)體中不允許出現(xiàn)多個(gè)重復(fù)的類(lèi)型
注意:如果結(jié)構(gòu)體里面有私有屬性也就是小寫(xiě)定義的字段,則不會(huì)被json使用
大家好,我是小白,有點(diǎn)黑的那個(gè)白。
最近遇到一個(gè)問(wèn)題,因?yàn)闃I(yè)務(wù)需求,需要對(duì)接第三方平臺(tái).
而三方平臺(tái)提供的一些HTTP(S)接口都有統(tǒng)一的密鑰生成規(guī)則要求.
為此我們封裝了一個(gè)獨(dú)立的包 xxx-go-sdk 以便維護(hù)和對(duì)接使用.
其中核心的部分是自定義HTTP Client,如下:
一些平臺(tái)會(huì)要求appKey/appSecret等信息,所以Client結(jié)構(gòu)體就變成了這樣,這時(shí)參數(shù)還比較少, 而且是必填的參數(shù),我們可以提供構(gòu)造函數(shù)來(lái)明確指定。
看起來(lái)很滿(mǎn)足,但是當(dāng)我們需要增加一個(gè) Timeout 參數(shù)來(lái)控制超時(shí)呢?
或許你會(huì)說(shuō)這還不簡(jiǎn)單,像下面一樣再加一個(gè)參數(shù)唄
那再加些其他的參數(shù)呢?那構(gòu)造函數(shù)的參數(shù)是不是又長(zhǎng)又串,而且每個(gè)參數(shù)不一定是必須的,有些參數(shù)我們又會(huì)考慮默認(rèn)值的問(wèn)題。
為此,勤勞但尚未致富的 gophers 們使用了總結(jié)一種實(shí)踐模式
首先提取所有需要的參數(shù)到一個(gè)獨(dú)立的結(jié)構(gòu)體 Options,當(dāng)然你也可以用 Configs 啥的.
然后為每個(gè)參數(shù)提供設(shè)置函數(shù)
這樣我們就為每個(gè)參數(shù)設(shè)置了獨(dú)立的設(shè)置函數(shù)。返回值 func(*Options) 看著有點(diǎn)不友好,我們提取下定義為單個(gè) Option 調(diào)整一下代碼
當(dāng)我們需要添加更多的參數(shù)時(shí),只需要在 Options 添加新的參數(shù)并添加新參數(shù)的設(shè)置函數(shù)即可。
比如現(xiàn)在要添加新的參數(shù) Timeout
這樣后續(xù)不管新增多少參數(shù),只需要新增配置項(xiàng)并添加獨(dú)立的設(shè)置函數(shù)即可輕松擴(kuò)展,并且不會(huì)影響原有函數(shù)的參數(shù)順序和個(gè)數(shù)位置等。
至此,每個(gè)選項(xiàng)是區(qū)分開(kāi)來(lái)了,那么怎么作用到我們的 Client 結(jié)構(gòu)體上呢?
首先,配置選項(xiàng)都被提取到了 Options 結(jié)構(gòu)體重,所以我們需要調(diào)整一下 Client 結(jié)構(gòu)體的參數(shù)
其次,每一個(gè)選項(xiàng)函數(shù)返回 Option,那么任意多個(gè)就是 ...Option,我們調(diào)整一下構(gòu)造函數(shù) NewClient 的參數(shù)形式,改為可變參數(shù),不再局限于固定順序的幾個(gè)參數(shù)。
然后循環(huán)遍歷每個(gè)選項(xiàng)函數(shù),來(lái)生成Client結(jié)構(gòu)體的完整配置選項(xiàng)。
那么怎么調(diào)用呢?對(duì)于調(diào)用方而已,直接在調(diào)用構(gòu)造函數(shù)NewClient()的參數(shù)內(nèi)添加自己需要的設(shè)置函數(shù)(WithXXX)即可
當(dāng)需要設(shè)置超時(shí)參數(shù),直接添加 WithTimeout即可,比如設(shè)置3秒的超時(shí)
配置選項(xiàng)的位置可以任意設(shè)置,不需要受常規(guī)的固定參數(shù)順序約束。
可以看到,這種實(shí)踐模式主要作用于配置選項(xiàng),利用函數(shù)支持的特性來(lái)實(shí)現(xiàn)的,為此得名 Functional Options Pattern,優(yōu)美的中國(guó)話(huà)叫做「函數(shù)選項(xiàng)模式」。
最后, 我們總結(jié)回顧一下在Go語(yǔ)言中函數(shù)選項(xiàng)模式的優(yōu)缺點(diǎn)
現(xiàn)在有個(gè)結(jié)構(gòu)體如下定義:
我們需要初始化結(jié)構(gòu)體,如果是其他語(yǔ)言,函數(shù)支持默認(rèn)參數(shù):
但是,go語(yǔ)言函數(shù)不支持默認(rèn)參數(shù),同時(shí)即使go語(yǔ)言支持默認(rèn)參數(shù),但是如果配置項(xiàng)過(guò)多,那么每一個(gè)配置項(xiàng)都得寫(xiě)一個(gè)默認(rèn)參數(shù),也不現(xiàn)實(shí)。
那么,在go語(yǔ)言中,我們?cè)趺磧?yōu)雅的給其初始化呢,這時(shí),就需要利用選項(xiàng)模式了(option)。
首先,我們定義一個(gè)option函數(shù)類(lèi)型:
它接收一個(gè)參數(shù): *Server 。
然后定義一個(gè) NewServer 函數(shù),它接收一個(gè) Option類(lèi)型的不定參數(shù):
最后,再直接定義一系列返回 Option的函數(shù)
使用時(shí),直接:
golang方法(method)返回值提取結(jié)構(gòu)體(struct)取不到地址的原因是,①返回值并沒(méi)有保存到變量中,返回值本身只是臨時(shí)保存在程序運(yùn)行的堆棧的某個(gè)不確定位置,不能取地址;②實(shí)參取地址用的操作符是是,而形參聲明變量類(lèi)型為指針,需要地址值用的才是*;③聲明形參為指針的參數(shù)的實(shí)參只能為地址值。
故先把修改后的代碼列出,修改要點(diǎn)是把“*NewPerson1().Speak()”改為“var b=NewPerson1();(b).Speak()”,同時(shí)把“NewPerson2().Speak()”改成“var a=NewPerson2();(a).Speak()”,代碼列出如下:
package main;
import "fmt";
type PersonA struct{
name string
}
func (p *PersonA) Speak () {
fmt.Println ( "person speak" ,p.name)
}
func (p PersonA) Walk ( ){
fmt . Println ( "person walk",p.name)}
func NewPerson1()(p PersonA){
return PersonA{"new Person1"}}
func NewPerson2()(p PersonA){
return PersonA{"new Person2"}}
func main () {
var a=NewPerson2 (); (a).Speak ();?
a .Walk ();
fmt. Println ("--------------------")?;
var b=NewPerson1 ();(b).Speak ();
b.Walk ()}
go代碼調(diào)試效果
關(guān)于指針變量的使用這一點(diǎn)go語(yǔ)言和其他有指針的程序語(yǔ)言如c語(yǔ)言是一樣的,從來(lái)只有返回值為地址/指針,而從沒(méi)有在賦值前給返回值取地址這種運(yùn)算,類(lèi)似的錯(cuò)誤晚點(diǎn)再整理。
不一樣的是,go語(yǔ)言更簡(jiǎn)單go語(yǔ)言函數(shù)可以使用結(jié)構(gòu)體或者結(jié)構(gòu)體的指針(pointer)以傳遞結(jié)構(gòu)體參數(shù),而且和c語(yǔ)言不一樣的是,go語(yǔ)言沒(méi)有區(qū)分結(jié)構(gòu)體指針和結(jié)構(gòu)體訪(fǎng)問(wèn)成員的運(yùn)算符,go語(yǔ)言只有“.”適用于兩種情況,而沒(méi)有c語(yǔ)言為結(jié)構(gòu)體指針專(zhuān)門(mén)準(zhǔn)備的“-”運(yùn)算符。
可以使用結(jié)構(gòu)體指針,作為結(jié)構(gòu)體的方法的參數(shù)以指代自身嗎,
作為C語(yǔ)言家族的一員,go和c一樣也支持結(jié)構(gòu)體。可以類(lèi)比于java的一個(gè)POJO。
在學(xué)習(xí)定義結(jié)構(gòu)體之前,先學(xué)習(xí)下定義一個(gè)新類(lèi)型。
新類(lèi)型 T1 是基于 Go 原生類(lèi)型 int 定義的新自定義類(lèi)型,而新類(lèi)型 T2 則是 基于剛剛定義的類(lèi)型 T1,定義的新類(lèi)型。
這里要引入一個(gè)底層類(lèi)型的概念。
如果一個(gè)新類(lèi)型是基于某個(gè) Go 原生類(lèi)型定義的, 那么我們就叫 Go 原生類(lèi)型為新類(lèi)型的底層類(lèi)型
在上面的例子中,int就是T1的底層類(lèi)型。
但是T1不是T2的底層類(lèi)型,只有原生類(lèi)型才可以作為底層類(lèi)型,所以T2的底層類(lèi)型還是int
底層類(lèi)型是很重要的,因?yàn)閷?duì)兩個(gè)變量進(jìn)行顯式的類(lèi)型轉(zhuǎn)換,只有底層類(lèi)型相同的變量間才能相互轉(zhuǎn)換。底層類(lèi)型是判斷兩個(gè)類(lèi)型本質(zhì)上是否相同的根本。
這種類(lèi)型定義方式通常用在 項(xiàng)目的漸進(jìn)式重構(gòu),還有對(duì)已有包的二次封裝方面
類(lèi)型別名表示新類(lèi)型和原類(lèi)型完全等價(jià),實(shí)際上就是同一種類(lèi)型。只不過(guò)名字不同而已。
一般我們都是定義一個(gè)有名的結(jié)構(gòu)體。
字段名的大小寫(xiě)決定了字段是否包外可用。只有大寫(xiě)的字段可以被包外引用。
還有一個(gè)點(diǎn)提一下
如果換行來(lái)寫(xiě)
Age: 66,后面這個(gè)都好不能省略
還有一個(gè)點(diǎn),觀(guān)察e3的賦值
new返回的是一個(gè)指針。然后指針可以直接點(diǎn)號(hào)賦值。這說(shuō)明go默認(rèn)進(jìn)行了取值操作
e3.Age 等價(jià)于 (*e3).Age
如上定義了一個(gè)空的結(jié)構(gòu)體Empty。打印了元素e的內(nèi)存大小是0。
有什么用呢?
基于空結(jié)構(gòu)體類(lèi)型內(nèi)存零開(kāi)銷(xiāo)這樣的特性,我們?cè)谌粘?Go 開(kāi)發(fā)中會(huì)經(jīng)常使用空 結(jié)構(gòu)體類(lèi)型元素,作為一種“事件”信息進(jìn)行 Goroutine 之間的通信
這種以空結(jié)構(gòu)體為元素類(lèi)建立的 channel,是目前能實(shí)現(xiàn)的、內(nèi)存占用最小的 Goroutine 間通信方式。
這種形式需要說(shuō)的是幾個(gè)語(yǔ)法糖。
語(yǔ)法糖1:
對(duì)于結(jié)構(gòu)體字段,可以省略字段名,只寫(xiě)結(jié)構(gòu)體名。默認(rèn)字段名就是結(jié)構(gòu)體名
這種方式稱(chēng)為 嵌入字段
語(yǔ)法糖2:
如果是以嵌入字段形式寫(xiě)的結(jié)構(gòu)體
可以省略嵌入的Reader字段,而直接訪(fǎng)問(wèn)ReaderName
此時(shí)book是一個(gè)各個(gè)屬性全是對(duì)應(yīng)類(lèi)型零值的一個(gè)實(shí)例。不是nil。這種情況在Go中稱(chēng)為零值可用。不像java會(huì)導(dǎo)致npe
結(jié)構(gòu)體定義時(shí)可以在字段后面追加標(biāo)簽說(shuō)明。
tag的格式為反單引號(hào)
tag的作用是可以使用[反射]來(lái)檢視字段的標(biāo)簽信息。
具體的作用還要看使用的場(chǎng)景。
比如這里的tag是為了幫助 encoding/json 標(biāo)準(zhǔn)包在解析對(duì)象時(shí)可以利用的規(guī)則。比如omitempty表示該字段沒(méi)有值就不打印出來(lái)。