声明方法的语法和声明普通函数非常类似,只是在函数名字前面加上一个参数,这个参数把这个方法绑定到它对应的类型上。
func (e Employee) ToString() (description string) {
return fmt.Sprintf("[%d, %s], from %s", e.ID, e.Name, e.Address)
}
(e Employee)EmployeeToString()Employeeethisself
在golang中,可以将一个方法绑定到任意的类型上(指针类型和接口类型除外),包括内置的基础类型,这也有些类似 C# 语言中的扩展方法。
继承type Employee struct {
ID int
Name, Address string
}
func (e Employee) ToString() (description string) {
return fmt.Sprintf("[%d, %s], from %s", e.ID, e.Name, e.Address)
}
func (e Employee) SayHello() string {
return fmt.Sprintf("hello, I'm %s", e.Name)
}
type EmployeeManager struct {
Employee // 匿名成员
ManagerLevel int
}
func (e EmployeeManager) ToString() (description string) {
return fmt.Sprintf("%s , level %d", e.Employee.ToString(), e.ManagerLevel)
}
var manager = EmployeeManager{
Employee: Employee{
ID: 2,
Name: "fooManager",
Address: "beijing",
},
ManagerLevel: 4,
}
fmt.Println(manager.SayHello())
manager.Employeemanager
ToStringEmployee.ToStringEmployeeManager.ToString
从这里可以看出,编译器在里面做了很多工作,导致在运行时分得特别清。我们不能将一个子类的实例赋值给父类的变量,所以想实现多态的话,需要借助接口,这个是编译器支持的,具体来讲就是:你声明一个接口类型的变量或参数,可以将任意一个实现了接口方法的类的实例赋值给它,在运行中也会执行对应的类的代码。
由于golang可以以匿名成员的方式实现继承的效果,那就意味着它可以实现多继承的效果。
方法接收者是指针类型的情况方法的类型也可以声明为类型的指针形式,比如:
func (e *Employee) ToString() (description string) {
return fmt.Sprintf("[%d, %s], from %s", e.ID, e.Name, e.Address)
}
&*
但是,与普通函数一样,如果你定义的接收者是枚举类型,那么在实际调用时会采用值传递的方式,也就意味着如果你修改了成员的值,原来的参数并不会发生变化。
func (e *EmployeeManager) Promote() {
e.ManagerLevel++
}
这里必须使用指针传递,否则成员的改动无效。
总结:指针类型或枚举类型,编译器都给予充分支持,怎么写都能行,但在运行时,指针类型会保留值的改动。
习惯:如果一个类型的任何一个方法使用了指针接受者,那么所有的方法都应该采用指针类型的接受者,即便有些只读的方法并不需要。
接收者的实参可以是没错,接收者可以是nil,golang的机制会将nil传递给方法并运行,至于会不会引发宕机异常,要看你有没有引用nil的引用。
func (e *Employee) SayHello() string {
if e == nil {
return "hello, I'm nobody"
}
return fmt.Sprintf("hello, I'm %s", e.Name)
}
var nilEmployee *Employee
fmt.Println(nilEmployee.SayHello()) // 输出 hello, I'm nobody
方法变量和方法表达式
这部分的目的是将一个方法像函数那样来调用。
toString := employee.ToString
fmt.Println(toString()) // func() string
fmt.Printf("%T\n", toString)
toString
promote := (*EmployeeManager).Promote
fmt.Printf("%T\n", promote) // func(*main.EmployeeManager)
promote(&manager)
fmt.Println(manager.ToString())
promote
变量.f()变量.fT.f(*T).f
这两个机制就实现了从方法到函数的转换。
golang中的封装golang只有一种控制可见性的机制,就是大写字母开头会导出,在包外可见,如果以小写字母开头,则在包内可见。
所以在golang中,对象的封装只能通过在结构体中定义小写字母开头的成员来实现,并且封装的单元是包,而不是类。
从封装的目的上看,封装主要是通过隐藏内部信息来降低外部使用它的复杂度,从这个意义上来讲,包内封装的必要性确实没那么大。