1、OOP
首先,Go 语言并不是面向对象的语言,只是可以通过一些方法来模拟面向对象。
1.1、封装
Go 语言是通过结构体(struct)来实现封装的。
1.2、继承
继承主要由下面这三种方式实现:
1.2.1、嵌套匿名字段
//Address 地址结构体 type Address struct { Province string City string } //User 用户结构体 type User struct { Name string Gender string Address //匿名字段 } func main() { var user2 User user2.Name = "小王子" user2.Gender = "男" user2.Address.Province = "山东" // 匿名字段默认使用类型名作为字段名 user2.City = "威海" // 匿名字段可以省略 fmt.Printf("user2=%#v\n", user2) //user2=main.User{Name:"小王子", Gender:"男", Address:main.Address{Province:"山东", City:"威海"}} }
1.2.2、嵌套结构体
//Address 地址结构体 type Address struct { Province string City string } //User 用户结构体 type User struct { Name string Gender string Address Address } func main() { user1 := User{ Name: "小王子", Gender: "男", Address: Address{ Province: "山东", City: "威海", }, } fmt.Printf("user1=%#v\n", user1)//user1=main.User{Name:"小王子", Gender:"男", Address:main.Address{Province:"山东", City:"威海"}} }
1.2.3、嵌套匿名结构体指针
//Animal 动物 type Animal struct { name string } func (a *Animal) move() { fmt.Printf("%s会动!\n", a.name) } //Dog 狗 type Dog struct { Feet int8 *Animal //通过嵌套匿名结构体实现继承 } func (d *Dog) wang() { fmt.Printf("%s会汪汪汪~\n", d.name) } func main() { d1 := &Dog{ Feet: 4, Animal: &Animal{ //注意嵌套的是结构体指针 name: "乐乐", }, } d1.wang() //乐乐会汪汪汪~ d1.move() //乐乐会动! }
而既然结构体可以继承,那么结构体就必须有方法,Go 语言的方法必须在方法名前面声明调用者。子类可以重写父类方法:如果在子结构体(或任何类型)上定义了一个与父结构体中同名的方法,那么这个方法就会覆盖父结构体中的方法。这就实现了重写。
1.3、多态
多态:一个事物拥有多种形态就是多态!
有多态就必须要有接口,因为接口就是为了解决多态这个问题的:
1.3.1、接口
- Go 语言提供了接口数据类型
- 接口就是把一些共性的方法放在一起定义
- Go 语言中的接口是隐式声明的(相比较 Java 会用 implements 关键字显示声明)
- 只有实现类把接口的方法全部实现才算实现了这个接口
接口的实现类都拥有多态的特性,因为它除了是自己还是它的接口类型。
package main import "fmt" // 接口 type USB interface { input() output() } // 结构体 type Mouse struct { name string } // 实现接口:实现了接口的所有方法才算实现了这个接口 func (mouse Mouse) input(){ fmt.Println(mouse.name,"鼠标输入") } func (mouse Mouse) output(){ fmt.Println(mouse.name,"鼠标输出") } type KeyBoard struct { name string } func (keyBoard KeyBoard) input(){ fmt.Println(keyBoard.name,"键盘输入") } func (keyBoard KeyBoard) output(){ fmt.Println(keyBoard.name,"键盘输出") } func test(u USB) { u.input() u.output() } func main() { mouse := Mouse{name: "罗技"} test(mouse) keyBoard := KeyBoard{name: "艾石头"} test(keyBoard) // 通过接口创建子类实例 var usb USB = Mouse{name: "外星人"} usb.input() // 但是接口是无法使用实现类的属性的 }
运行结果:
罗技 鼠标输入 罗技 鼠标输出 艾石头 键盘输入 艾石头 键盘输出 外星人 鼠标输入
1.3.2、空接口
空接口不包含任何方法,所以所有的结构体都默认实现了空接口(类似于 Java 的 Object)!
所谓的空接口,就是:
type 接口名称 interface{}
go 语言中的 any 其实就是空接口,我们可以在源码中看到:
如果我们定义一个方法或者函数它可以传入一个空接口类型,那么就相当于任何类型都可以传入这个方法或函数,因为任何结构体类型的都实现了空接口。比如我们 go 语言中的打印方法的参数就都是 any ... 。
2、接口
上面只是描述了接口是怎么实现多态的,但是对接口的用法并没有深入介绍,这里我们详细介绍接口的用法。
2.1、接口的定义
type 接口名 interface{ 方法名1(参数列表) (返回值列表) 方法名2(参数列表) (返回值列表) // ... }
需要注意的是:
- 接口类型名:Go语言的接口在命名时,一般会在单词后面添加er,如有写操作的接口叫Writer,有关闭操作的接口叫closer等。接口名最好要能突出该接口的类型含义。
- 方法名:当方法名首字母是大写且这个接口类型名首字母也是大写时,这个方法可以被接口所在的包(package)之外的代码访问。
- 参数列表、返回值列表:参数列表和返回值列表中的参数变量名可以省略。
2.2、接口类型变量
所谓的接口类型变量就就像 Java 中的:
Map<String,Integer> map; HashMap<String,Integer> map1 = new HashMap<>(); TreeMap<String,Integer> map2 = new TreeMap<>(); map = m1; map = m2;
这里的变量 map 就是一个接口变量,接口变量可以通过任何实现类来赋值。
2.3、接口的嵌套
Go 语言中的接口可以组合嵌套,这是区别于 Java 很大的一点。在 Go标准库 io 源码中就有很多接口之间互相组合的示例:
// src/io/io.go type Reader interface { Read(p []byte) (n int, err error) } type Writer interface { Write(p []byte) (n int, err error) } type Closer interface { Close() error } // ReadWriter 是组合Reader接口和Writer接口形成的新接口类型 type ReadWriter interface { Reader Writer } // ReadCloser 是组合Reader接口和Closer接口形成的新接口类型 type ReadCloser interface { Reader Closer } // WriteCloser 是组合Writer接口和Closer接口形成的新接口类型 type WriteCloser interface { Writer Closer }
同时,接口也可以作为结构体的字段,就像 Java 中 Map 可以作为对象属性一样:
// src/sort/sort.go // Interface 定义通过索引对元素排序的接口类型 type Interface interface { Len() int Less(i, j int) bool Swap(i, j int) } // reverse 结构体中嵌入了Interface接口 type reverse struct { Interface }
2.4、类型断言
类型断言就像 Java 中的强转一样,一般是把一个抽象的接口类型转为一个确定的实现类型。好像说我们可以"断言"这个接口类型一定是这个实现类类型。
2.4.1、语法
x.(T)
- x:表示接口类型的变量(如果不是接口类型的就在前面加上空接口)
- T:表示断言 x 是 T 类型
注意:类型断言的返回结果是两个参数,第一个返回值是一个转为断言类型后的变量,第二个返回值是转为断言的结果(布尔类型,代表成功/失败)
对于数值类型( 比如 int、string、float64... )这些不是接口类型的数据,如果要做类型断言就需要给它前面加个空接口,因为所有类型都是隐式地实现了空接口的。
str := "10" // 第2个返回值是断言结果 res,_ := interface{}(str).(int) fmt.Println(res) // 10
对于接口类型变量,如果我们能知道它是哪个实现类型就可以直接进行类型断言:
var usb USB = Mouse{name: "外星人"} // 类型断言 这里没有接收第二个返回值,代表丢弃 m := usb.(Mouse) fmt.Println(m)
上面的 USB 是接口类型,而它的地址指向一个 Mouse 类型的实例,所以我们可以断言这个 USB 实例一定是 Mouse 类型。