返回顶部
首页 > 资讯 > 后端开发 > GO >Go基础教程系列之Go接口使用详解
  • 153
分享到

Go基础教程系列之Go接口使用详解

2024-04-02 19:04:59 153人浏览 泡泡鱼
摘要

接口用法简介 接口(interface)是一种类型,用来定义行为(方法)。 type Namer interface { my_method1() my_method

接口用法简介

接口(interface)是一种类型,用来定义行为(方法)。

type Namer interface {
    my_method1()
    my_method2(para)
    my_method3(para) return_type
    ...
}

但这些行为不会在接口上直接实现,而是需要用户自定义的方法来实现。所以,在上面的Namer接口类型中的方法my_methodN都是没有实际方法体的,仅仅只是在接口Namer中存放这些方法的签名(签名 = 函数名+参数(类型)+返回值(类型))。

当用户自定义的类型实现了接口上定义的这些方法,那么自定义类型的值(也就是实例)可以赋值给接口类型的值(也就是接口实例)。这个赋值过程使得接口实例中保存了用户自定义类型实例。

例如:

package main

import (
	"fmt"
)

// Shaper 接口类型
type Shaper interface {
	Area() float64
}

// Circle struct类型
type Circle struct {
	radius float64
}

// Circle类型实现Shaper中的方法Area()
func (c *Circle) Area() float64 {
	return 3.14 * c.radius * c.radius
}

// Square struct类型
type Square struct {
	length float64
}

// Square类型实现Shaper中的方法Area()
func (s *Square) Area() float64 {
	return s.length * s.length
}

func main() {
	// Circle类型的指针类型实例
	c := new(Circle)
	c.radius = 2.5

	// Square类型的值类型实例
	s := Square{3.2}

	// Sharpe接口实例ins1,它自身是指针类型的
	var ins1 Shaper
	// 将Circle实例c赋值给接口实例ins1
	// 那么ins1中就保存了实例c
	ins1 = c
	fmt.Println(ins1)

	// 使用类型推断将Square实例s赋值给接口实例
	ins2 := s
	fmt.Println(ins2)
}

上面将输出:

&{2.5}
{3.2}

从上面输出结果中可以看出,两个接口实例ins1和ins2被分别赋值后,分别保存了指针类型的Circle实例c和值类型的Square实例s

另外,从上面赋值ins1和ins2的赋值语句上看:

ins1 = c
ins2 := s

是否说明接口实例ins就是自定义类型的实例?实际上接口是指针类型(指向什么见下文)。这个时候,自定义类型的实例c、s称为具体实例,ins实例是抽象实例,因为ins接口中定义的行为(方法)并没有具体的行为模式,而c、s中的行为是具体的。

因为接口实例ins也是自定义类型的实例,所以当接口实例中保存了自定义类型的实例后,就可以直接从接口上调用它所保存的实例的方法。例如:

fmt.Println(ins1.Area())   // 输出19.625
fmt.Println(ins2.Area())   // 输出10.24

这里ins1.Area()调用的是Circle类型上的方法Area(),ins2.Area()调用的则是Square类型上的方法Area()。这说明Go的接口可以实现面向对象中的多态:可以按需调用名称相同、功能不同的方法

接口实例中存的是什么

前面说了,接口类型是指针类型,但是它到底存放了什么东西?

接口类型的数据结构是2个指针,占用2个机器字长。

当将类型实例c赋值给接口实例ins1后,用println()函数输出ins1和c,比较它们的地址:

println(ins1)
println(c)

输出结果:

(0x4ceb00,0xc042068058)
0xc042068058

从结果中可以看出,接口实例中包含了两个地址,其中第二个地址和类型实例c的地址是完全相同的。而第二个地址c是Circle的指针类型实例,所以ins中的第二个值也是指针。

ins中的第一个是指针是什么?它所指向的是一个内部表结构iTable,这个Table中包含两部分:第一部分是实例c的类型信息,也就是*Circle,第二部分是这个类型(Circle)的方法集,也就是Circle类型的所有方法(此示例中Circle只定义了一个方法Area())。

所以,如图所示:

注意,上图中的实例c是指针,是指针类型的Circle实例。

对于值类型的Square实例s,ins2保存的内容则如下图:

实际上接口实例中保存的内容,在反射(reflect)中体现的淋漓尽致,reflect所有的一切都离不开接口实例保存的内容。

方法集(Method Set)规则

官方手册对Method Set的解释:https://golang.org/ref/speC#Method_sets

实例的method set决定了它所实现的接口,以及通过receiver可以调用的方法。

方法集是类型的方法集合,对于非接口类型,每个类型都分两个Method Set:值类型实例是一个Method Set,指针类型的实例是另一个Method Set。两个Method Set由不同receiver类型的方法组成:

实例的类型       receiver
--------------------------------------
 值类型:T       (T Type)
 指针类型:*T    (T Type)或(T *Type)

也就是说:

  • 值类型的实例的Method Set只由值类型的receiver(T Type)组成
  • 指针类型的实例的Method Set由值类型和指针类型的receiver共同组成,即(T Type)(T *Type)

这是什么意思呢?从receiver的角度去考虑:

receiver        实例的类型
---------------------------
(T Type)        T 或 *T
(T *Type)       *T

上面的意思是:

  • receiver是指针类型的方法只可能存在于指针类型的实例方法集中
  • receiver是值类型的方法既存在于值类型的实例方法集中,也存在于指针类型的方法集中

从实现接口方法的角度上看:

  • 如果某类型实现接口的方法的receiver是(T *Type)类型的,那么只有指针类型的实例*T才算是实现了这个接口,因为这个方法不在值类型的实例T方法集中
  • 如果某类型实现接口的方法的receiver是(T Type)类型的,那么值类型的实例T和指针类型的实例*T都算实现了这个接口,因为这个方法既在值类型的实例T方法集中,也在指针类型的实例*T方法集中

举个例子。接口方法Area(),自定义类型Circle有一个receiver类型为(c *Circle)的Area()方法时,说明实现了接口的方法,但只有Circle实例的类型为指针类型时,这个实例才算是实现了接口,才能赋值给接口实例,才能当作一个接口参数。如下:

package main

import "fmt"

// Shaper 接口类型
type Shaper interface {
	Area() float64
}

// Circle struct类型
type Circle struct {
	radius float64
}

// Circle类型实现Shaper中的方法Area()
// receiver类型为指针类型
func (c *Circle) Area() float64 {
	return 3.14 * c.radius * c.radius
}

func main() {
	// 声明2个接口实例
	var ins1, ins2 Shaper

	// Circle的指针类型实例
	c1 := new(Circle)
	c1.radius = 2.5
	ins1 = c1
	fmt.Println(ins1.Area())

	// Circle的值类型实例
	c2 := Circle{3.0}
	// 下面的将报错
	ins2 = c2
	fmt.Println(ins2.Area())
}

报错结果:

cannot use c2 (type Circle) as type Shaper
in assignment:
        Circle does not implement Shaper (Area method has
pointer receiver)

它的意思是,Circle值类型的实例c2没有实现Share接口的Area()方法,它的Area()方法是指针类型的receiver。换句话说,值类型的c2实例的Method Set中没有receiver类型为指针的Area()方法

所以,上面应该改成:

ins2 = &c2

再声明一个方法,它的receiver是值类型的。下面的代码一切正常。

type Square struct{
    length float64
}

// 实现方法Area(),receiver为值类型
func (s Square) Area() float64{
    return s.length * s.length
}

func main() {
    var ins3,ins4 Shaper

    // 值类型的Square实例s1
    s1 := Square{3.0}
    ins3 = s1
    fmt.Println(ins3.Area())

    // 指针类型的Square实例s2
    s2 := new(Square)
    s2.length=4.0
    ins4 = s2
    fmt.Println(ins4.Area())
}

所以,从struct类型定义的方法的角度去看,如果这个类型的方法有指针类型的receiver方法,则只能使用指针类型的实例赋值给接口变量,才算是实现了接口。如果这个类型的方法全是值类型的receiver方法,则可以随意使用值类型或指针类型的实例赋值给接口变量。下面这两个对应关系,对于理解很有帮助:

实例的类型       receiver
--------------------------------------
 值类型:T       (T Type)
 指针类型:*T    (T Type)或(T *Type)
 
receiver        实例的类型
---------------------------
(T Type)        T 或 *T
(T *Type)       *T

很经常的,我们会直接使用推断类型的赋值方式(如ins2 := c2)将实例赋值给一个变量,我们以为这个变量是接口的实例,但实际上并不一定。正如上面值类型的c2赋值给ins2,这个ins2将是从c2数据结构拷贝而来的另一个副本数据结构,并非接口实例,但这时通过ins2也能调用Area()方法:

c2 = Circle{3.2}
ins2 := c2
fmt.Println(ins2.Area())  // 正常执行

之所以能调用,是因为Circle类型中有Area()方法,但这不是通过接口去调用的。

所以,在使用接口的时候,应当尽量使用var先声明接口类型的实例,再将类型的实例赋值给接口实例(如var ins1,ins2 Shaper),或者使用ins1 := Shaper(c1)的方式。这样,如果赋值给接口实例的类型实例没有实现该接口,将会报错。

但是,为什么要限制指针类型的receiver只能是指针类型的实例的Method Set呢?

看下图,假如指针类型的receiver可以组成值类型实例的Method Set,那么接口实例的第二个指针就必须找到值类型的实例的地址。但实际上,并非所有值类型的实例都能获取到它们的地址。

哪些值类型的实例找不到地址?最常见的是那些简单数据类型的别名类型,如果匿名生成它们的实例,它们的地址就会被Go彻底隐藏,外界找不到这个实例的地址。

例如:

package main

import "fmt"

type myint int

func (m *myint) add() myint {
	return *m + 1
}
func main() {
	fmt.Println(myint(3).add())
}

以下是报错信息:找不到myint(3)的地址

abc\abc.go:11:22: cannot call pointer method on myint(3)
abc\abc.go:11:22: cannot take the address of myint(3)

这里的myint(3)是匿名的myint实例,它的底层是简单数据类型int,myint(3)的地址会被彻底隐藏,只会提供它的值对象3。

普通方法和实现接口方法的区别

对于普通方法,无论是值类型还是指针类型的实例,都能正常调用,且调用时拷贝的内容都由receiver的类型决定

func (T Type) method1   // 值类型receiver
func (T *Type) method2  // 指针类型receiver

指针类型的receiver决定了无论是值类型还是指针类型的实例,都拷贝实例的指针。值类型的receiver决定了无论是值类型还是指针类型的实例,都拷贝实例本身

所以,对于person数据结构:

type person struct {}
p1 := person{}       // 值类型的实例
p2 := new(person)    // 指针类型的实例

p1.method1()p2.method1()都是拷贝整个person实例,只不过Go对待p2.method1()时多一个"步骤":将其解除引用。所以p2.method1()等价于(*p2).method1()

p1.method2()p2.method2()都拷贝person实例的指针,只不过Go对待p1.method2()时多一个"步骤":创建一个额外的引用。所以,p1.method2()等价于(&p1).method2()

而类型实现接口方法时,method set规则决定了类型实例是否实现了接口。

receiver        实例的类型
---------------------------
(T Type)        T 或 *T
(T *Type)       *T

对于接口abc、接口方法method1()、method2()和结构person:

type abc interface {
	method1
	method2
}

type person struct {}
func (T person) method1   // 值类型receiver
func (T *person) method2  // 指针类型receiver

p1 := abc(person)  // 接口变量保存值类型实例
p2 := abc(&person) // 接口变量保存指针类型实例

p2.method1()p2.method2()以及p1.method1()都是允许的,都会通过接口实例去调用具体person实例的方法。

p1.method2()是错误的,因为method2()的receiver是指针类型的,导致p1没有实现接口abc的method2()方法。

接口类型作为参数

将接口类型作为参数很常见。这时,那些实现接口的实例都能作为接口类型参数传递给函数/方法。

例如,下面的myArea()函数的参数是n Shaper,是接口类型。

package main

import (
	"fmt"
)

// Shaper 接口类型
type Shaper interface {
	Area() float64
}

// Circle struct类型
type Circle struct {
	radius float64
}

// Circle类型实现Shaper中的方法Area()
func (c *Circle) Area() float64 {
	return 3.14 * c.radius * c.radius
}

func main() {
	// Circle的指针类型实例
	c1 := new(Circle)
	c1.radius = 2.5
	myArea(c1)
}

func myArea(n Shaper) {
	fmt.Println(n.Area())
}

上面myArea(c1)是将c1作为接口类型参数传递给n,然后调用c1.Area(),因为实现了接口方法,所以调用的是Circle的Area()。

如果实现接口方法的receiver是指针类型的,但却是值类型的实例,将没法作为接口参数传递给函数,原因前面已经解释过了,这种类型的实例没有实现接口。

以接口作为方法或函数的参数,将使得一切都变得灵活且通用,只要是实现了接口的类型实例,都可以去调用它。

用的非常多的fmt.Println(),它的参数也是接口,而且是变长的接口参数:

$ go doc fmt Println
func Println(a ...interface{}) (n int, err error)

每一个参数都会放进一个名为a的Slice中,Slice中的元素是接口类型,而且是空接口,这使得无需实现任何方法,任何东西都可以丢到fmt.Println()中来,至于每个东西怎么输出,那就要看具体情况:由类型的实现的String()方法决定。

接口类型的嵌套

接口可以嵌套,嵌套的内部接口将属于外部接口,内部接口的方法也将属于外部接口。

例如,File接口内部嵌套了ReadWrite接口和Lock接口。

type ReadWrite interface {
    Read(b Buffer) bool
    Write(b Buffer) bool
}
type Lock interface {
    Lock()
    Unlock()
}
type File interface {
    ReadWrite
    Lock
    Close()
}

除此之外,类型嵌套时,如果内部类型实现了接口,那么外部类型也会自动实现接口,因为内部属性是属于外部属性的。

更多关于Go基础教程系列之Go接口的使用方法请查看下面的相关链接

您可能感兴趣的文档:

--结束END--

本文标题: Go基础教程系列之Go接口使用详解

本文链接: https://lsjlt.com/news/146242.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Go基础教程系列之Go接口使用详解
    接口用法简介 接口(interface)是一种类型,用来定义行为(方法)。 type Namer interface { my_method1() my_method...
    99+
    2024-04-02
  • Go基础教程系列之WaitGroup用法实例详解
    正常情况下,新激活的goroutine(协程)的结束过程是不可控制的,唯一可以保证终止goroutine(协程)的行为是main goroutine(协程)的终止。也就是说,我们并不...
    99+
    2024-04-02
  • Go基础教程系列之defer、panic和recover详解
    defer关键字 defer关键字可以让函数或语句延迟到函数语句块的最结尾时,即即将退出函数时执行,即便函数中途报错结束、即便已经panic()、即便函数已经return了,也都会执...
    99+
    2024-04-02
  • Go基础教程系列之回调函数和闭包详解
    Go回调函数和闭包 当函数具备以下两种特性的时候,就可以称之为高阶函数(high order functions): 函数可以作为另一个函数的参数(典型用法是回调函数)函数可以返回另...
    99+
    2024-04-02
  • Go基础教程系列之数据类型详细说明
    每一个变量都有数据类型,Go中的数据类型有: 简单数据类型:int、float、complex、bool和string数据结构或组合(composite):struct、array、...
    99+
    2024-04-02
  • Go语言基础go接口用法示例详解
    目录概述语法定义接口实现接口空接口接口的组合总结 概述 Go 语言中的接口就是方法签名的集合,接口只有声明,没有实现,不包含变量。 语法 定义接口 type [接口名] inte...
    99+
    2024-04-02
  • Go基础系列:Go切片(分片)slice详解
    slice表示切片(分片),例如对一个数组进行切片,取出数组中的一部分值。在现代编程语言中,slice(切片)几乎成为一种必备特性,它可以从一个数组(列表)中取出任意长度的子数组(列...
    99+
    2024-04-02
  • java基础教程之接口
    定义:接口就是多个类的共有规范(里面的抽象方法),是一种引用数据类型。小提示:基本数据类型包括数值型(整数和浮点数)、字符型、布尔型。格式:public interface 接口名称{ //接口内容 }备注:接口.java编译后仍然是接口...
    99+
    2019-04-11
    java入门 java 接口
  • Go基础教程系列之import导入包(远程包)和变量初始化详解
    import导入包 搜索路径 import用于导入包: import ( "fmt" "net/http" "mypkg" ) 编译器会根据上面指定的相对路...
    99+
    2024-04-02
  • android基础教程之context使用详解
    在android中有两种context,一种是application context,一种是activity context,通常我们在各种类和方法间传递的是activity ...
    99+
    2022-06-06
    context 教程 Android
  • Go语言基础学习之Context的使用详解
    目录前言基本用法Context控制goroutine的生命周期使用 WithValue() 传递数据使用 WithCancel() 取消操作使用 WithDeadline() 设置截...
    99+
    2023-05-19
    Go语言Context使用 Go语言Context用法 Go Context
  • Go语言基础之Time包详解
    Time包是Go语言中用于处理时间的一个标准库。它提供了一系列函数和类型,用于获取当前时间、时间格式化、时间计算等操作。在Go语言中...
    99+
    2023-08-29
    Go语言
  • Kotlin 基础教程之类、对象、接口
    Kotlin 基础教程之类、对象、接口Kotlin中类、接口相关概念与Java一样,包括类名、属性、方法、继承等,如下示例:interface A { fun bar() fun foo() { // 可选方法体 }}class C...
    99+
    2023-05-31
    kotlin 对象
  • Go语言基础学习之数组的使用详解
    目录1. Array(数组)2. 声明数组3. 数组初始化3.1 方式一3.2 方式二3.3 方式三3.4 多维数组4. 遍历数组&取值5. 数组拷贝和传参数组相必大家都很熟...
    99+
    2022-12-30
    Go语言数组使用 Go语言数组 Go 数组
  • Java基础学习之接口详解
    目录概述定义格式含有抽象方法含有默认方法和静态方法含有私有方法和私有静态方法基本的实现实现的概述抽象方法的使用默认方法的使用静态方法的使用私有方法的使用接口的多实现抽象方法默认方法静...
    99+
    2022-11-13
    Java接口使用 Java接口
  • 使用GO语言接口实现日志记录:教程详解
    日志记录是每个应用程序都需要的一个重要功能。它可以帮助开发人员及时发现并解决应用程序中的问题,提高应用程序的稳定性。在GO语言中,我们可以通过实现一个接口来实现日志记录功能。 下面是一个简单的例子,演示如何使用GO语言接口实现日志记录。 ...
    99+
    2023-06-25
    教程 接口 日志
  • Go语言基础go install命令使用示例详解
    目录go install一、使用二、包名和目录名的关系三、注意 go install 编译并安装代码包,对于库,会生成目标库文件,并且放置到GOPATH/pgk目录下。 对于可执文件...
    99+
    2024-04-02
  • Go语言基础go fmt命令使用示例详解
    go fmt 命令主要是用来帮你格式化所写好的代码文件【很多第三方集成软件都是使用了go fmt命令】 一、使用: go fmt <文件名>.go 使用go fmt命令...
    99+
    2024-04-02
  • CobaltStrike使用教程详解(基础)
    声明:本文仅限学习研究讨论,切忌做非法乱纪之事! 大家好,今天简单来聊聊CobaltStrike,这是我们后渗透阶段必不可少的神器。 Cobalt Strike 是一款流行的渗透测试工具,广泛用于红队操作和渗透测试。它由Raphael Mu...
    99+
    2023-09-13
    php 开发语言
  • Go语言基础学习之指针详解
    目录1. 什么是指针2. 指针地址 & 指针类型3. 指针取值4. 空指针5. make6. new7. make 和 new 的区别8. 问题今天来说说 Go 语言基础中的...
    99+
    2022-12-30
    Go语言指针使用 Go语言指针 Go 指针
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作