在 Go 语言中,函数和方法不太一样,有明确的概念区分。其他语言中,比如 PHP 函数就是方法,方法
就是函数,但在 Go 语言中,函数是不属于任何结构体、类型的方法,也就是说函数是没有接收者的;而方法
是有接收者的,我们所说的方法要么属于一个结构体的,要么属于一个新定义的类型的
函数
函数和方法,虽然概念不同,但是定义非常相似。函数的定义声明没有接收者,如下示例:
package main import "fmt" // 定义一个求两数之和的函数 func add(a,b int) int { return a + b } func main() { sum := add(1,2) fmt.Println(sum) }
上面的例子中,我们定义了一个 add 函数,它的函数签名是 func add(a,b int) int ,没有接收者,直接定义在 go
的一个包之下,可以直接用,例子中的这个函数名是小写的 add,所以它的作用域只有当前包,不能被其他包使用,
如果我们把函数名以大写开头,可以被其他包调用,这也是 Go 中大小写的作用
方法
方法的声明和函数类似,她们的区别是:方法在定义的时候在 func 和 方法名之间有一个参数,这个参数就是方
法的接收者,这样我们定义的这个方法就和接收者绑定在一起了吧,称之为这个接收者的方法:
// 定义一个结构体 type person struct { name string } func (p person) String() string { return "the person name is " + p.name }
上面的这个例子中,func 和 方法名 String 之间的参数(p person) 就是接收者,现在我们说,类型 person 有了一个方法
String,现在看下如何使用:
func main() { p := person{name : "张三"} fmt.Println(p.String()) }
Go 语言中接收者分为两种类型,值接受者和指针接收者。我们上面的例子中,就是值类型接收者的示例
使用值类型接收者定义的方法,在调用的时候,使用的其实就是值接受者的一个副本,所以对该值的任何操作,不会影响
原来的值
package main import "fmt" type person struct { name string } func (p person) String() string { return "the person name is " + p.name } func (p person) modify() { p.name = "李四" } func main() { p := person{"张三"} // 值类型接收者 p.modify() fmt.Println(p.String()) }
上面的例子中,打印出来的值是 "张三",对其进行的修改无效。如果我们使用一个指针作为接收者,那么就会起作用了,
因为指针接收者传递的是一个指向原值指针的拷贝,指针的副本,指向的还是原来类型的值,所以修改时,同时也会影响原
来类型变量的值
package main import "fmt" type person struct { name string } func (p person) String() string { return "the person name is " + p.name } func (p *person) modify() { p.name = "李四" } func main() { p := person{"张三"} // 值类型接收者 p.modify() fmt.Println(p.String()) }
在调用方法的时候,传递的接收者本质上都是副本,只不过一个是值副本,另一个是指向这个值指针的副本。指针具有
指向原值的特性,所以修改了指针指向的值,也就修改了原有的值。我们可以简单的理解为值接收者使用的是值的副本来调用
方法,而指针接收者使用的是实际值调用方法
上面的例子中我们发现,在调用指针接收者方法的时候,使用的也是一个值的变量,并不是一个指针,修改如下:
p := person{"张三"} (&p).modify()
这样也是可以的,如果我们没有强制使用指针进行调用,Go 编译器会帮我们取指针,同样的,如果是一个值接收者的方法
使用指针也可以调用,Go 编译器会自动解引,如下:
p := person{"张三"} fmt.Println((&p).String())
所以,方法的调用既可以是值也可以是指针
多值返回
Go 语言支持函数方法的多值返回,也就是说我们定义的函数方法支持可以返回多个值,比如标准库里的很多方法,都是返回两
个值,第一个是函数需要返回的值,第二个是出错时返回的错误信息
package main import ( "fmt" "log" "os" ) func main() { file , err := os.Open("/usr/tmp") if err != nil { log.Fatal(err) return } fmt.Println(file) }
如果返回的值,我们不想使用,可以使用 _ 进行忽略:
file , _ := os.Open("/usr/tmp/")
多个值返回的定义也非常简单,如下示例:
func add(a,b int) (int,error) { return a + b, nil }
函数方法声明定义的时候,采用逗号分隔,因为是多个返回,还要用括号扩起来,返回的值还是使用 return 关键字,以逗号分隔,和
声明的顺序一致
可变参数
函数方法的参数可以是任意多个,这种我们称之为可变参数,比如我们常用的 fmt.Println() 这类函数,可以接收可变参数
func main() { fmt.Println("1","2","3","4") }
自己定义一个可接收可变参数的函数,如果可变参数的类型是一样的则可以使用省略号 ... 代替:
func print(a ...interface{}) { for _,v := range a{ fmt.Println(v) } }
可变参数本质上是一个数组,所以我们可以像数组一样使用它