Golang函數(shù)這些神操作你知道哪些
「1. 用type關鍵字可以定義函數(shù)類型,函數(shù)類型變量可以作為函數(shù)的參數(shù)或返回值。」
package?main
import?"fmt"
func?add(a,?b?int)?int?{
?return?a?+?b
}
func?sub(a,?b?int)?int?{
?return?a?-?b
}
type?Task?func(int,?int)?int
func?exec(t?Task,?a,?b?int)?int?{
?return?t(a,?b)
}
func?main()?{
?a?:=?exec(add,?10,?20)
?fmt.Println(a)
?b?:=?exec(sub,?100,?95)
?fmt.Println(b)
}
解析:type Task func(int, int) int,這句是說,使用type關鍵字定義1個類型為func的Task,這個func有2個int形參、1個int返回值。再看exec這個函數(shù),它有3個形參,形參t的類型是剛定義的函數(shù)類型Task,另外兩個你懂的,我就不說了。
「2. 匿名函數(shù)的玩法是真的騷,看看騷在哪里」
栗子1:匿名函數(shù)可以直接賦給變量
func?main()?{
?var?aaa?=?func(a,?b?int)?int?{
??return?a?+?b
?}
?ret?:=?aaa(89,?78)
?fmt.Println(ret)
}
輸出:
167
栗子2:匿名函數(shù)作為函數(shù)入?yún)?/p>
這個例子感覺跟剛才使用type關鍵字定義函數(shù)類型的例子有點相同的趕腳,仔細一看,好像也只是在使用的方式上是一樣,定義函數(shù)的套路又是不同的。
func?work(f?func(int,?int)?int,?a,?b?int)?int?{
?return?f(a,?b)
}
func?add(a,?b?int)?int?{
?return?a?+?b
}
func?main()?{
?a?:=?work(add,?100,?200)
?fmt.Println(a)
}
輸出:
300
解析:形參f的類型就是匿名函數(shù),繼續(xù)看func(int, int) int,這個匿名函數(shù)接收兩個int形參,返回值也是int類型。另外兩個形參a和b,也是int,想必你懂了,我就不廢話了。
栗子3:匿名函數(shù)作為函數(shù)出參
匿名函數(shù)作為函數(shù)出參(作為函數(shù)返回值),經(jīng)不斷調測,有3種騷包玩法
騷包玩法1
//?先來個簡單的
func?work()?func()?int?{
?return?func()?int?{
??return?10?+?20
?}
}
func?main()?{
?f?:=?work()
?ret?:=?f()
?fmt.Println(ret)
}
輸出:
30
騷包玩法2
//?再改造下
func?work(a,?b?int)?func()?int?{
?return?func()?int?{
??return?a?+?b
?}
}
func?main()?{
?f?:=?work(500,?20)
?ret?:=?f()
?fmt.Println(ret)
}
輸出:
520
騷包玩法3
//?再次改造
func?work()?func(int,?int)?int?{
?return?func(a,?b?int)?int?{
??return?a?+?b
?}
}
func?main()?{
?f?:=?work()
?ret?:=?f(600,?50)
?fmt.Println(ret)
}
輸出:
650
對3種騷包玩法的簡單解析:
再這里就解析一下第3種玩法,能搞懂這個玩法,前面2個玩法,你就自然懂了。work函數(shù)沒有入?yún)ⅲ怯谐鰠ⅲㄒ部梢哉f是返回值),出參是匿名函數(shù)func(int, int) int,這個定義在返回值里的匿名函數(shù)有2個int類型的形參(入?yún)ⅲ┖?個int類型的返回值。再看看函數(shù)體內部,沒干很復雜活兒,而是直接return了匿名函數(shù),函數(shù)體里的這個匿名函數(shù)是和定義在返回值里的類型保持一致的,也是接收了2個int的形參a和b,返回值也是int,啥也沒干,就做了個相加。好郁悶!不知道你看懂了沒?再看看是如何使用work函數(shù)的,關鍵點就在這里,調用work()賦給了變量f,這時候,f它就是一個函數(shù)了,再調用f(600, 50),想必你已經(jīng)知道了為啥要傳入2個int值,這下搞定!不知道把你繞暈沒,這玩法確實很騷。
「3. 匿名函數(shù)和延遲調用」
延遲調用的規(guī)則是:按照「先進后出」的順序,也就是說函數(shù)返回前是會被執(zhí)行的,而且是按照先進后出的順序。如何起到延遲的效果,是需要注冊的,可通過defer關鍵字進行注冊。那么什么場景下需要用到延遲調用呢?比如常見的場景:當一個函數(shù)被即將執(zhí)行完后,也就是完成任務的最后一刻,需要回收和釋放某些資源。
延遲調用的機制可以配合匿名函數(shù)來使用,這樣就可以讓匿名函數(shù)被直接調用,只能說是真的騷。
先看個小栗子:
package?main
import?"fmt"
func?work()?bool?{
?fmt.Println("函數(shù)開始工作...")
?defer?func()?{
??fmt.Println("回收相關資源工作開始!")
?}()
?defer?func()?{
??fmt.Println("清理工作開始!")
?}()
?fmt.Println("函數(shù)正在工作...")
?fmt.Println("函數(shù)工作完畢...")
?return?true
}
func?main()?{
?status?:=?work()
?fmt.Println(status)
}
輸出:
函數(shù)開始工作...
函數(shù)正在工作...
函數(shù)工作完畢...
清理工作開始!
回收相關資源工作開始!
true
在上面的栗子中,定義了個work函數(shù),先看看輸出的結果,感受下。在work函數(shù)中,注冊了兩個延遲調用,work函數(shù)從開始->正在->完畢,結束后才執(zhí)行defer注冊的匿名函數(shù),這里要著重注意延遲調用規(guī)則:「先進后出」,也就是先注冊后執(zhí)行。“回收相關資源工作”是先注冊的,他的執(zhí)行順序排在了“清理工作”的后面。感受到了嗎?
最后說明一下:defer關鍵字讓匿名函數(shù)實現(xiàn)了可直接調用,那么使用defer注冊延遲調用時要注意,defer關鍵字的后面一定是要函數(shù)或方法的調用,不能直接寫語句哦。
到此這篇關于Golang函數(shù)這些神操作你知道哪些的文章就介紹到這了,更多相關Golang函數(shù)內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Golang中字符串(string)與字節(jié)數(shù)組([]byte)一行代碼互轉實例
golang語言本身就是c的工具集,開發(fā)c的程序用到的大部分結構體,內存管理,攜程等,golang基本都有,下面這篇文章主要給大家介紹了關于Golang中字符串(string)與字節(jié)數(shù)組([]byte)一行代碼互轉的相關資料,需要的朋友可以參考下2022-09-09

