有两个理由使用指针接收器
- 第一个是为了让方法可以修改它的接收器指向的值。
- 第二种方法是避免在每次方法调用时复制值。例如,如果接收方是一个大的结构体,这可能会更有效率。
通常,给定类型上的所有方法都应该具有值或指针接收器,但不能两者混合。(我们将在接下来的几页中看到原因。)
接口
接口是定义的一组方法签名。
实现接口必须啊实现接口的所有方法。
实现接口必须要值接收器方法。
在底层,一个接口值可以被认为是一个值和具体类型的元组
(value, type)
调用接口值上的方法将在其底层类型上执行同名方法。
如果接口内部的具体值本身是nil,该方法将用nil接收器调用。(但还是会调用方法,因为有类型,可以在方法内通过判断接收器为nil 来避免空指针错误)
package main
import "fmt"
type I interface {
M()
}
type T struct {
S string
}
func (t *T) M() {
if t == nil {
fmt.Println("<nil>")
return
}
fmt.Println(t.S)
}
func main() {
var i I
var t *T
i = t
describe(i)
i.M()
i = &T{"hello"}
describe(i)
i.M()
}
func describe(i I) {
fmt.Printf("(%v, %T)\n", i, i)
}
注意,持有nil具体值的接口值本身是非nil的。
nil接口值既不包含值也不包含具体类型。
在nil接口上调用方法是一个运行时错误,因为在接口元组中没有类型来指示要调用哪个具体方法。
package main
import "fmt"
type I interface {
M()
}
func main() {
var i I
describe(i)
i.M()
}
func describe(i I) {
fmt.Printf("(%v, %T)\n", i, i)
}
// =========================
(<nil>, <nil>)
panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xc0000005 code=0x0 addr=0x0 pc=0x49c106]
goroutine 1 [running]:
main.main()
指定0个方法的接口类型称为空接口
interface{}
空接口可以保存任何类型的值。(每种类型至少实现0个方法。)
空接口由处理未知类型值的代码使用。例如,fmt.Print接受 interface{} 类型的任意数量的参数。
package main
import "fmt"
func main() {
var i interface{}
describe(i)
i = 42
describe(i)
i = "hello"
describe(i)
}
func describe(i interface{}) {
fmt.Printf("(%v, %T)\n", i, i)
}
// =========================
(<nil>, <nil>)
(42, int)
(hello, string)