golang学习笔记22-面向对象(四):接口【重要】

本节也是GO核心部分,很重要。
注:由于导包语句已经在19讲(笔记19:面向对象的引入)展示过了,所以这里就不展示了。

一、定义与实现

(1)接口中可以定义一组方法,但不需要实现,不需要方法体。并且接口中不能包含任何变量。到某个自定义类型要使用的时候(实现接口的时候),再根据具体情况把这些方法具体实现出来
(2)实现接口要实现所有的方法才算实现
(3)Golang不需要显式的实现接口。Golang中没有implement关键字,Golang中实现接口是基于方法的,不是基于接口的,例如:
A接口a,b方法
B接口a,b方法
C结构体实现了a,b方法,那么C实现了A接口,也可以说实现了B接口,只要实现全部方法即可,和实际接口耦合性很低,比Java松散得多。
(4)接口目的是为了定义规范,具体由别人来实现即可。
示例,首先在utils中定义接口,结构体及方法:

//接口的定义:定义一组规则或规范,由别人实现
type SayHello interface {
	Say() //声明没有实现的方法
}

//接口的实现:1.定义类型,这里是结构体
type Chinese struct {
}
type American struct {
}

//2.实现接口的方法(具体实现)
func (c Chinese) Say() {
	fmt.Println("你好")
}

func (a American) Say() {
	fmt.Println("hello")
}

//定义一个函数,专门用于各国人打招呼,接收(被实现的)接口变量
func Greet(s SayHello) {
	s.Say()
}

然后在main中调用:

func main() {
	c := utils.Chinese{}
	a := utils.American{}
	utils.Greet(c)
	utils.Greet(a)
}

可能有些人看到我每次调用utils觉得麻烦,那么这里就提一个问题:如果用别名,比如定义utils.Chinese为Chinese,utils.Greet(c)还有效吗?答案是否定的,因为编译器不认为这两个是同一类型,自然也就不认为新的类型实现了接口的方法,所以别名定义后,新类型也必须实现接口的方法!

二、一些细节

【1】接口本身不能创建实例,比如在Greet中添加以下代码会有警告:

var ss SayHello
ss.Say()

警告:nil dereference in dynamic method call,这表示ss是空引用,空引用是无法进行任何操作的,所以运行原来main中的代码一定会报错。
但是接口可以指向一个实现了该接口的变量,比如在main中添加如下代码:

var s utils.SayHello = c
s.Say()

这里s接收的不再是空引用,而是实现了接口的结构体,所以能调用接口的方法,注意该类型(结构体)一定要实现方法,不实现没法赋值。
【2】只要是自定义数据类型,就可以实现接口,不仅仅是结构体类型。示例,uitils中的定义为:

type MyInt int

func (i MyInt) Say() {
	fmt.Println("hello", i)
}

main中的调用为:

func main() {
	var i utils.MyInt = 10
	i.Say()
}

【3】一个自定义类型可以实现多个接口,示例:
uitils中的定义为:

type InterA interface {
	A1()
}
type InterB interface {
	B1()
}
type Stu struct {
}

func (s Stu) A1() {
	fmt.Println("run A1")
}
func (s Stu) B1() {
	fmt.Println("run B1")
}

main中的调用为:

func main() {
	s := utils.Stu{}
	s.A1()
	s.B1()
}

【4】一个接口可以继承多个接口,这时如果要实现该接口,必须将继承接口的方法都实现。示例,uitils中的定义为:

type InterC interface {
	InterA
	InterB
	C1()
}

func (s Stu) C1() {
	fmt.Println("run C1")
}

main中的调用为:

func main() {
	s := utils.Stu{}
	s.A1()
	s.B1()
	s.C1()
}

【5】空接口可以接收除了接口之外的所有类型(接口不能赋值接口),可以理解为所有类型都实现了空接口。由于这个比较简单,我就直接在main中写了,用到了可变参数,涉及的知识点丰富:

func printInf(inf ...interface{}) {
	for _, value := range inf {
		fmt.Printf("%T ", value)
	}
}
func main() {
	// 整,浮,字符串,布,数组,函数,指针,切片,映射,结构体
	intg, float, str, b, arr := 1, 1.1, "1", true, [1]int{1}
	pointer, slice, fun := new(int), arr[:], func() {}
	mapp, stct := make(map[string]int, 2), utils.Stu{}

	// 创建一个切片包含所有变量
	inf := []interface{}{intg, float, str, b, arr, pointer, slice, fun, mapp, stct}

	// 传递切片作为参数,注意加上...将切片展开
	printInf(inf...)
}

程序输出为:

int
float64
string
bool
[1]int
*int
[]int
func()
map[string]int
utils.Stu
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术卷

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值