Go 语言基础之面向对象编程

发布于:2024-05-07 ⋅ 阅读:(31) ⋅ 点赞:(0)

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 类型。