本文为转载,原文:Golang 学习笔记(02)—— 函数
函数定义
函数
是每一门开发语言的最基础的部件,下面看下go语言中的函数是怎么定义的:
func Add(a, b int)int{
return a + b
}
func
是定义函数的关键字,Add
是函数名称,int
是返回值,小括号内是函数的参数。可以随意安排函数定义的顺序,go在编译时会扫描所以的文件。
多返回值
go中函数支持多返回值,可以返回任意数量的返回值。多值返回在Go语言中是经常被用到的,比如,一个函数同时返回结果和异常。
下面我们看一个例子,divide
函数是计算a/b的结果,返回商和余数。
package main
import "fmt"
func main(){
quotient, remainder := divide(5, 3)
fmt.Println("商为:", quotient, "余数为:", remainder)
}
func divide(a, b int)(int, int){
quotient := a / b
remainder := a % b
return quotient, remainder
}
变参函数
go中的函数支持变参,变参就是说函数可以有任意数量的参数。变参本质上就是一个slice
,且必须是最后一个参数
。将slice
传递给变参函数时,注意用...
展开,否则就当做单个参数处理了。
请看以下例子:
package main
import "fmt"
func main(){
result := sum(3,5,7,9)
fmt.Println("结果为:", result)
}
func sum(aregs ...int) int {
s := 0
for _, number := range aregs{
s += number
}
return s
}
至于什么是
slice
,range
又是什么鬼,在后面讲到数组的数组的时候详细介绍,slice
就相当于数组,range
类似一个循环,循环每个字元素的key, value。另外其中的_
表示不接受改返回值。
defer
defer是go语言所特有的,defer的作用是延迟执行
,在函数返回前,按照栈的形式后进先出
的原则一次执行每个defer注册的函数。这样可以保证函数在返回前被调用,通常用来进行资源释放,错误的处理,清理数据等。下面是一个读文件的例子。
package main
import "fmt"
import "os"
func main(){
str, err := readFile("main.go")
if err != nil{
fmt.Println(err.Error())
return
}
fmt.Println(str)
}
func readFile(strFileName string)(string, error){
f, err := os.Open(strFileName)
if err != nil{
fmt.Println("文件读取失败")
return "", err
}
defer f.Close()
buf := make([]byte, 1024)
var strContent string = ""
for{
n, _ := f.Read(buf)
if n == 0{
break
}
strContent += string(buf[0:n])
}
return strContent, nil
}
函数类型
函数也是一种类型,拥有相同的参数,相同的返回值的函数,是同一种类型。用type
来定义函数类型。下面例子中display
函数输出大于5的数值。
package main
import "fmt"
type MyFuncType func(int) bool
func isBigThan5(n int)bool{
return n > 5
}
func display(arr []int, f MyFuncType){
for _, v := range arr{
if f(v){
fmt.Println(v)
}
}
}
func main(){
arr := []int{1,2,3,4,5,6,7,8,9}
display(arr, isBigThan5)
}
在上面的例子中,
type MyFuncType func(int) bool
定义了一个函数类型,将其命名为MyFuncType
,接受一个int类型的参数,并返回一个bool类型的结果。isBigThan5
是MyFuncType
类型的函数,函数类型。跟C里的函数指针有点像,他是可以把函数当做参数传入到另一个函数里,也有点像委托。
错误处理
go语言中没有try...catch...finally...
这种结构化异常处理,而是用panic
代替throw
跑出异常。使用recover
函数来捕获异常。Recover
仅在defer
函数中使用才能捕获到异常,此时函数的执行流程已经中断,无法恢复到后续位置继续执行。
package main
import "fmt"
func main(){
test()
}
func test(){
defer func (){
if err := recover(); err != nil{
fmt.Println(err)
}
}()
divide(5,0)
fmt.Println("end of test")
}
func divide(a, b int) int{
return a / b
}
完
本文为原创,转载请注明出处