Golang-接口(interface)

接口(interface)

接口是一种类型

最初的intstringbool,再到稍微复杂的ArrayMapSlice。他们都称之为基础数据类型,以及到多维度符合类型的结构体。以及今日咱们所需要学习的接口

在Go语言编程中,Go(强类型语言),也就是说必须是一种具体的类型,当我们需要只关注能调用它的什么方法,而不关注它是什么类型,该怎么办呢?

Go语言中为了解决类似上面的问题,就设计了接口这个概念。接口区别于我们之前所有的具体类型,接口是一种抽象的类型。当你看到一个接口类型的值时,你不知道它是什么,唯一知道的是通过它的方法能做什么。

疑问:只关心调用的函数,而不关注其类型
package main

import (
	"fmt"
)

type person struct{}
type dog struct{}

func (p person) speak() {
	fmt.Println("shit~")
}

func (d dog) speak() {
	fmt.Println("汪汪汪~")
}

func do() {
	// 接受一个参数,进来什么,什么就要speak
	x.speak()
}
func main() {

}

接口的定义

type 接口类型名 interface{
    方法名1( 参数列表1,参数列表2 ... ) (返回值列表1,返回值列表2 ...)
    方法名2( 参数列表1,参数列表2 ...) (返回值列表1,返回值列表2 ...)
   	... ...
}
  • 接口名:使用type将接口定义为自定义的类型名。Go语言的接口在命名时,一般会在单词后面添加er,如有写操作的接口叫Writer,有字符串功能的接口叫Stringer等。接口名最好要能突出该接口的类型含义。
  • 方法名:当方法名首字母是大写且这个接口类型名首字母也是大写时,这个方法可以被接口所在的包(package)之外的代码访问。
  • 参数列表、返回值列表:参数列表和返回值列表中的参数变量名可以省略。

那么为了解决以上问题,我们可以定义接口。实现如下

package main

import (
	"fmt"
)

// 接口
type speak interface {
	speak()
}

// 结构体
type person struct{}
type dog struct{}

// 结构体person的实现
func (p person) speak() {
	fmt.Println("shit~")
}

// 结构体dog 实现
func (d dog) speak() {
	fmt.Println("汪汪汪~")
}

func do(s speak) {
	// 接受一个参数,进来什么,什么就调用它的speak
	s.speak()
}
func main() {
	var p1 person
	var d1 dog

	do(p1)
	do(d1)
}
// shit~
// 汪汪汪~

实现接口的条件

一个变量如果实现了接口中全部的方法,那么此变量就实现了这个接口。

接口是一个需要实现的类型(方法列表)

package main

import (
	"fmt"
)

// 接口
type speak interface {
	speak()
}

// 结构体
type person struct{}
type dog struct{}

// 结构体person的实现
func (p person) speak() {
	fmt.Println("shit~")
}

// 结构体dog 实现
func (d dog) speak() {
	fmt.Println("汪汪汪~")
}

func do(s speak) {
	// 接受一个参数,进来什么,什么就调用它的speak
	s.speak()
}
func main() {
	var p1 person
	var d1 dog

	// 定义一个接口类型:speak的变量speaks
	var speaks speak
	speaks = d1
	speaks = p1
	fmt.Print(speaks)
}
// {}

接口类型变量

接口类型变量能够存储所有实现了该接口的实例。

package main

import "fmt"

type say interface {
	say()
}

type cats struct{}
type dogs struct{}

func (c cats) say() {
	fmt.Println("Fish~")
}
func (d dogs) say() {
	fmt.Print("Shit~")
}
func sayer(s say) {
	// 接受一个参数,进来什么,什么就调用它的speak
	s.say()
}
func main() {
	var x say
	a := cats{}
	b := dogs{}
	x = a
	x.say()
	x = b
	x.say()
}

值的接受者与指针接收者实现接口

值的接受者实现接口
package main

import "fmt"

type moving interface {
	move()
}
type dog struct{}
type cat struct{}

func (d dog) move() {
	fmt.Println("丁丁~")
}

func (c cat) move() {
	fmt.Println("喵呜~")
}
func move(m moving) {
	// 接受一个参数,进来什么,什么就调用它的speak
	m.move()
}
func main() {
	var x moving
	a := dog{}
	b := &cat{}
	x = a
	x.move()
	x = b
	x.move()
}

从上面的代码中我们可以发现,使用值接收者实现接口之后,不管是dog结构体还是结构体指针*dog类型的变量都可以赋值给该接口变量。因为Go语言中有对指针类型变量求值的语法糖,cat指针x内部会自动求值(* ** x)

指针接收者实现接口

同样的代码我们再来测试一下使用指针接收者有什么区别:

package main

import "fmt"

type moving interface {
	move()
}
type dog struct{}
type cat struct{}

func (d dog) move() {
	fmt.Println("丁丁~")
}

func (c *cat) move() {
	fmt.Println("喵呜~")
}
func move(m moving) {
	// 接受一个参数,进来什么,什么就调用它的speak
	m.move()
}
func main() {
	var x moving
	a := dog{} // a是dog类型
	x = a      // 可以接收dog类型
	x.move()
	b := cat{}
	x = b // 不可以接受指针类型
	x.move()
}
// # command-line-arguments
// ./pointer.go:28:4: cannot use b (type cat) as type moving in assignment:
// cat does not implement moving (move method has pointer receiver)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值