Go(函数)

四、函数

4.1声明函数

格式:

func functionname(parametername type) returntype {    
   // 函数体(具体实现的功能)  
}  

多参数,参数类型简写,如果多个参数变量,而且是同类型则可以简写,只需在最后一个参数后添加该类型

func calculateBill(price, no int) int {    
    var totalPrice = price * no // 商品总价 = 商品单价 * 数量  
    return totalPrice // 返回总价  
}   

多返回值
如果一个函数有多个返回值,那么这些返回值必须用 ( 和 ) 括起来。

func rectProps(length, width float64)(float64, float64) {  
     var area = length * width
     var perimeter = (length + width) * 2
     return area, perimeter
}

命名返回值

func rectProps(length, width float64)(area, perimeter float64) {  
    area = length * width
    perimeter = (length + width) * 2
    return // 不需要明确指定返回值,默认返回 area, perimeter 的值
}

调用函数

result := add(1,2)

Go语言中传入和返回参数在调用和返回时都是使用值传递,这里需要注意的是指针、切片和map等引用型对象指向的内容在参数传递中不会发生复制。

空白符
_ 在 Go 中被用作空白符,可以用作表示任何类型的任何值,用来跳过不要的计算结果。

func main() {  
    area, _ := rectProps(10.8, 5.6) // 返回值周长被丢弃
    fmt.Printf("Area %f ", area)
}

4.2 函数变量

Go语言中函数也是一种类型,可以和其他类型一样被保存在变量中。

func fire(){}

func main(){
    //声明一个函数变量
    var f func();
    //赋值
    f = fire
    //调用f相当于调用fire
    f()
}

字符串的链式处理。
需要字符串切片和处理链。

//字符串切片
list := []string{"shaokan1","shaokan2","shaokan3","shaokan4"}

//处理函数链
chain := []func(string)string{
    strings.TrimSpace,
    //自带函数
    strings.ToUpper,
    //自定义函数
    removePrefix,
}

4.3 匿名函数

匿名函数没有函数名,只有函数体,往往以变量的方式被传递。
格式:func(参数列表)(返回参数列表){函数体}

//最后(100)表示对匿名函数进行调用,传递参数为100
func (data int){
    fmt.Println(data)
}(100)


//匿名函数赋值给变量

f := func (data int){
    fmt.Println(data)
}
//调用
f(100)

//匿名函数作为回调函数
func visit(list []string , f func(string)){
    for _,v = range list{
        f(v)
    }
}

func main(){
    visit([]{"1","2","3"},func(a){
        fmt.Println(a)
    })
}

4.4 闭包 (Closure)——引用了外部变量的匿名函数

闭包是引用了自由变量的函数,被引用的自由变量和函数一同存在,即使已经离开了自由变量的环境也不会被释放或者删除,在闭包中可以继续使用这个自由变量。
闭包具有记忆效应。
闭包被用于实现工厂模式的生成器。

//生成器
func playerMake(name string) func()(string int){
    hp := 150
    //返回闭包
    return func()(string int){
        return name,hp
    }
}

func main(){
    //创建一个生成器
    ge := playerMake("shaokan")
    //执行闭包,获取数据
    name , hp := ge()
    
}

4.5 可变参数 —— 参数数量不固定的函数形式

格式:
func 函数名(固定参数, v...T)(返回参数){函数体}
v 为可变参数变量,类型为[]T,也就是拥有多个T元素的T类切片,v和T之间有...组成;T为可变参数的类型,当T为interface{} 时,传入的可以是任意类型。
fmt相关函数的声明如下:

//Println 
func Println(a ...interface{})(n int, err error){
    return Fprintln(os.Stdout , a...)
}
//实现了使用时,传入的值类型不受限制。
fmt.Println(5 ,"hello",&struct{a int }{1} , true)


//Printf
func Printf(format string ,a ...interface{})(n,int , err, error){
    return Fprintln(os.Stdout , a...)
}
//Print使用时,第一个参数必须传入字符串类型值
fmt.Printf("value : %v %f",true , 32.3)

遍历可变参数列表

func jonsStrings(slist ...string) string{
    //定义一个字节缓冲
    var b bytes.Buffer
    //将遍历出的字符串连续写入字节数组
    for _,s := range slist{
        b.WriteString(s)
    }
    
    return b.String()
}

获取参数类型,如需,通过switch去获得数据类型。

for _ , s := range alist{
    var typeString string
    //类型断言
    sitch s.(type){
    case bool:
        typeString = "bool"
    case string:
        typeString = "string"
        .
        .
        .
    }
}

在多个可变参数函数中传递参数

func rawPrint(rawList ...interface{}){
    ...
}

func print(slist ...interface){
    //传递可变参数,使用变量...
    rawPrint(slist...)
}

4.6 延迟执行语句 (defer)

在defer 归属的函数即将返回时,将延迟处理的语句按defer的逆序进行执行。函数结束可以是正常返回也可以是发生宕机。

func main(){
    defer fmt.Println(1)
    defer fmt.Println(2)
    defer fmt.Println(3)
    
    //输出结果会是 3 2 1的顺序
}

用法:在函数退出时释放资源
1.并发解锁
2.释放文件句柄 (defer f.close)

4.7 处理运行时发生的错误

两个特征:
1.一个可能造成错误的函数,需要返回一个错误接口(error)。
2.在函数调用后检查错误,如果发生错误,进行必要的错误处理。

自定义一个错误

var err = errors.New("this is an error")

代码中使用,类似以下格式,代码设计上需要想想

//定义除数为0的错误
var errDivisionByZero = errors.New("division by zero")
func div(dividend , divisor int)(int , error){
    if(divisor == 0){
        return 0 , errDivisionByZero
    }
    
    ...
}

4.8 宕机(panic)

直接使用 panic() 引起程序崩溃
配合defer可用于宕机后要处理的事情

4.9 宕机恢复(recover)

panic 和 recover 的关系
1.有panic没有 recover ,程序宕机。
2.有panic 有 recover ,程序不会宕机。执行完对应的defer后,从宕机点退出当前函数后继续执行。

你可能感兴趣的:(Go(函数))