函式是基于功能或者邏輯進行聚合的可復用的代碼塊,將一些復雜的、冗長的代碼抽離封裝成多個代碼片段,即函式,有助于提高代碼邏輯的可讀性和可維護性,不同于Python,由于 Go lang是編譯型語言,編譯之后再運行,所以函式的定義順序無關痛癢,
函式宣告
在 Go lang里,函式宣告語法如下:
func function_name(parameter_list) (result_list) {
//函式邏輯
}
這里使用function的簡寫形式 func關鍵詞,后面依次接 function_name(函式名) , parameter_list(引數串列) , result_list(回傳值串列)以及函式體 ,
parameter_list(引數串列)成員:函式的引數名以及引數型別,這些引數作為區域變數,其值由引數呼叫者提供,函式中的引數串列和回傳值并非是必須的,
result_list(回傳值串列):函式回傳值的變數名以及型別,如果函式回傳一個無名變數或者沒有回傳值,回傳值串列的括號是可以省略的,
如果有連續若干個引數的型別一致,那么只需在最后一個引數后添加該型別:
package main
import "fmt"
// 函式回傳一個無名變數,回傳值串列的括號省略
func sum(x int, y int) int {
return x + y
}
// 無引數串列和回傳值
func printBookName() {
fmt.Println("go lang1.18")
}
// 引數的型別一致,只在最后一個引數后添加該型別
func sub(x, y int) int {
return x - y
}
func main() {
fmt.Println("1 + 1 = ", sum(1, 1))
printBookName()
fmt.Println("2 - 1 =", sub(2, 1))
}
程式回傳:
command-line-arguments
1 + 1 = 2
go lang1.18
2 - 1 = 1
不定長引數
和Python一樣,Go lang也支持不定長引數,即引數有多少個并不確定的情況,
在引數型別前面加 ... 表示一個切片,用來接收呼叫者傳入的引數,注意,如果該函式下有其他型別的引數,這些其他引數必須放在引數串列的前面,切片必須放在最后:
package main
import "fmt"
func show(args ...string) int {
sum := 0
for _, item := range args {
fmt.Println(item)
sum += 1
}
return sum
}
func main() {
fmt.Println(show("1", "2", "3"))
}
和Python的*args用法差不多,但需要注意必須要宣告函式的資料型別,程式回傳:
1
2
3
3
如果傳多個引數的資料型別都不一樣,可以指定型別為 ...interface{} ,然后再進行遍歷:
package main
import "fmt"
func PrintType(args ...interface{}) {
for _, arg := range args {
switch arg.(type) {
case int:
fmt.Println(arg, "type is int.")
case string:
fmt.Println(arg, "type is string.")
case float64:
fmt.Println(arg, "type is float64.")
default:
fmt.Println(arg, "is an unknown type.")
}
}
}
func main() {
PrintType(1, 3.1415, "go lang 1.18")
}
此外,還可以使用 ... 可以用來解序列,能將函式的可變引數(即切片)一個一個取出來,傳遞給另一個可變引數的函式,而不是傳遞可變引數變數本身:
package main
import "fmt"
func main() {
var s []string
s = append(s, []string{"1", "2", "3"}...)
fmt.Println(s)
}
這里將字串切片取出來后,傳遞給內置的append方法,程式回傳:
[1 2 3]
函式的回傳值
一個函式可以沒有回傳值,也可以有一個回傳值,也可以有回傳多個值:
package main
import "fmt"
func swap(x, y string) (string, string) {
return y, x
}
func SumAndProduct(A, B int) (add int, Multiplied int) {
add = A + B
Multiplied = A * B
return
}
func main() {
a, b := swap("Mahesh", "Kumar")
fmt.Println(a, b)
fmt.Println(SumAndProduct(1, 2))
}
程式回傳:
Kumar Mahesh
3 2
_ 是Go lang里的空白識別符號,它可以代替任何型別的任何值,我們可以利用它來忽略某些別人會用到但我們不會用到的函式回傳值:
package main
import (
"fmt"
)
func rectProps(length, width float64) (float64, float64) {
var area = length * width
var perimeter = (length + width) * 2
return area, perimeter
}
func main() {
area, _ := rectProps(10.8, 5.6) // perimeter is discarded
fmt.Printf("Area %f ", area)
}
程式回傳:
Area 60.480000
匿名函式
有點類似Python中的lambda運算式,但實際上并不是作為語法糖而存在:
package main
import (
"fmt"
)
func main() {
f := func() {
fmt.Println("hello world")
}
f() //hello world
fmt.Printf("%T\n", f) //列印 func()
}
程式回傳:
hello world
func()
一望而知,只是匿名而已,但通過變數可呼叫,另外也可以擁有引數:
package main
import (
"fmt"
)
func main() {
f:=func(args string){
fmt.Println(args)
}
f("hello world")//hello world
//或
(func(args string){
fmt.Println(args)
})("hello world")//hello world
//或
func(args string) {
fmt.Println(args)
}("hello world") //hello world
}
程式回傳:
hello world
hello world
hello world
基本上,匿名函式和命名函式用法上并無二致,
閉包(closure)
很多語言都有閉包的概念,簡單理解就是函式的嵌套:
package main
import "fmt"
func main() {
a := Fun()
b:=a("hello ")
c:=a("hello ")
fmt.Println(b)//worldhello
fmt.Println(c)//worldhello hello
}
func Fun() func(string) string {
a := "world"
return func(args string) string {
a += args
return a
}
}
程式回傳:
worldhello
worldhello hello
這里我們將方法作為引數傳遞到方法內部執行,這樣內層的函式可以使用外層函式的所有變數,即使外層函式已經執行完畢,
延遲函式
延遲其實是延遲(defer)陳述句,延遲陳述句被用于執行一個函式呼叫,在這個函式之前,延遲陳述句回傳:
package main
import "fmt"
func main() {
a := 1
b := 2
defer fmt.Println(b)
fmt.Println(a)
}
程式回傳:
1
2
說白了就是一種倒裝的形式,非延遲陳述句先執行,最后再執行延遲陳述句,
延遲也并不僅僅局限于函式內部陳述句,延遲一個方法呼叫也是可以的:
package main
import (
"fmt"
)
type person struct {
firstName string
lastName string
}
func (p person) fullName() {
fmt.Printf("%s %s", p.firstName, p.lastName)
}
func main() {
p := person{
firstName: "go lang",
lastName: "python",
}
defer p.fullName()
fmt.Printf("Welcome ")
}
程式回傳:
Welcome go lang python
初始化函式
顧名思義,和Python中的魔法方法init一樣,可以提前做一些初始化操作:
package main
import "fmt"
var a int = initVar()
func init() {
fmt.Println("init2")
}
func init() {
fmt.Println("init")
}
func initVar() int {
fmt.Println("init var...")
return 100
}
func main() {
fmt.Println("main...")
}
程式回傳:
init var...
init2
init
這里的初始化順序是:變數初始化->init()->main()
和Python不同的是,每個包可以有多個初始化函式,
結語
歸根結底,函式可以被認為是Go lang中的一種資料型別,可以作為另一個函式的引數,也可以作為另一個函式的回傳值,使用起來相當靈活,但我們也不能矯枉過正,毫無節制地用函式封裝邏輯,造成過度封裝的現象,
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/501604.html
標籤:其他
