我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Go接口怎么使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Go接口怎么使用

本篇内容主要讲解“Go接口怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Go接口怎么使用”吧!

接口用法简介

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

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

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

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

例如:

package mainimport ("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中就保存了实例cins1 = cfmt.Println(ins1)// 使用类型推断将Square实例s赋值给接口实例ins2 := sfmt.Println(ins2)}

上面将输出:

&{2.5}{3.2}

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

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

ins1 = cins2 := s

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

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

fmt.Println(ins1.Area())   // 输出19.625fmt.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())。

所以,如图所示:

Go接口怎么使用

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

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

Go接口怎么使用

实际上接口实例中保存的内容,在反射(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 mainimport "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.5ins1 = c1fmt.Println(ins1.Area())// Circle的值类型实例c2 := Circle{3.0}// 下面的将报错ins2 = c2fmt.Println(ins2.Area())}

报错结果:

cannot use c2 (type Circle) as type Shaperin assignment:        Circle does not implement Shaper (Area method haspointer 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 := c2fmt.Println(ins2.Area())  // 正常执行

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

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

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

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

Go接口怎么使用

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

例如:

package mainimport "fmt"type myint intfunc (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   // 值类型receiverfunc (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 {method1method2}type person struct {}func (T person) method1   // 值类型receiverfunc (T *person) method2  // 指针类型receiverp1 := abc(person)  // 接口变量保存值类型实例p2 := abc(&person) // 接口变量保存指针类型实例

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

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

接口类型作为参数

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

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

package mainimport ("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.5myArea(c1)}func myArea(n Shaper) {fmt.Println(n.Area())}

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

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

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

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

$ go doc fmt Printlnfunc 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接口怎么使用”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Go接口怎么使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Go接口怎么使用

本篇内容主要讲解“Go接口怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Go接口怎么使用”吧!接口用法简介接口(interface)是一种类型,用来定义行为(方法)。type Namer
2023-06-30

Go接口怎么用

这篇文章给大家分享的是有关Go接口怎么用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。接口是一种抽象类型,是对其他类型行为的概括与抽象,从语法角度来看,接口是一组方法定义的集合。很多面向对象的语言都有接口这个概念
2023-06-27

Go语言中的接口怎么使用

今天小编给大家分享一下Go语言中的接口怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。接口在 Go 语言中,接口是一种
2023-07-04

TypeScript接口怎么使用

这篇文章主要讲解了“TypeScript接口怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“TypeScript接口怎么使用”吧!前言:TS的核心原则之一就是对所具有的结构 进行类型检
2023-06-22

java.sql.ResultSetMetaData接口怎么使用

java.sql.ResultSetMetaData接口是用于描述ResultSet中的元数据的接口。它提供了一些方法来获取和操作ResultSet的元数据信息。使用步骤如下:1. 首先,需要获取ResultSet对象。可以通过执行SQL查
2023-10-23

java.sql.ResultSetMetaData接口怎么使用

Java.sql.ResultSetMetaData接口ResultSetMetaData接口提供有关ResultSet对象中列的元数据信息,包括列名、数据类型、大小和可空性。使用此接口,开发人员可以在JDBC中获取数据库元数据,从而了解从数据库检索的数据详细信息,有利于动态查询、错误处理和性能优化。
java.sql.ResultSetMetaData接口怎么使用
2024-04-11

怎么使用API接口

这篇文章主要介绍“怎么使用API接口”,在日常操作中,相信很多人在怎么使用API接口问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么使用API接口”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!接口交互前
2023-06-16

go怎么声明空接口

在Go语言中,可以使用`interface{}`来声明空接口。下面是一个示例代码:package mainimport "fmt"func main() {var emptyInterface interface{}emptyInterf
2023-10-25

C++中怎么使用接口

C++中怎么使用接口,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。首先我们需要一些宏:// // Interfaces.h // #define Interface
2023-06-17

GO语言中接口和接口型函数如何使用

这篇文章主要讲解了“GO语言中接口和接口型函数如何使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“GO语言中接口和接口型函数如何使用”吧!// A Getter loads data fo
2023-07-05

Java Cloneable接口怎么使用

要使用Java中的Cloneable接口,您需要遵循以下步骤:1. 在您的类定义中实现Cloneable接口,例如:```javapublic class MyClass implements Cloneable {// 类定义}```2.
2023-08-18

GO语言中接口和接口型函数的具体使用

本文主要介绍了GO语言中接口和接口型函数的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-03-09

怎么使用C#的API接口

这篇文章主要介绍“怎么使用C#的API接口 ”,在日常操作中,相信很多人在怎么使用C#的API接口 问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么使用C#的API接口 ”的疑惑有所帮助!接下来,请跟
2023-06-03

C#的INotifyPropertyChanged接口怎么使用

这篇文章主要介绍“C#的INotifyPropertyChanged接口怎么使用”,在日常操作中,相信很多人在C#的INotifyPropertyChanged接口怎么使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对
2023-06-29

Node.js怎么使用express写接口

这篇文章主要讲解了“Node.js怎么使用express写接口”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Node.js怎么使用express写接口”吧!创建基本的服务器//导入expre
2023-07-05

怎么使用javascript调用c接口

今天小编给大家分享一下怎么使用javascript调用c接口的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。第一步:创建C接口
2023-07-06

Angular之ControlValueAccessor接口怎么使用

这篇文章主要讲解了“Angular之ControlValueAccessor接口怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Angular之ControlValueAccessor
2023-07-05

Java8中Function接口怎么使用

这篇文章主要介绍了Java8中Function接口怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java8中Function接口怎么使用文章都会有所收获,下面我们一起来看看吧。Java 8 中 Func
2023-07-02

怎么使用Remix写API接口

这篇文章主要介绍“怎么使用Remix写API接口”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么使用Remix写API接口”文章能帮助大家解决问题。接口种类普通 get/post api:即可满足
2023-07-06

java怎么使用HttpClient调用接口

这篇文章主要介绍“java怎么使用HttpClient调用接口”,在日常操作中,相信很多人在java怎么使用HttpClient调用接口问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java怎么使用HttpC
2023-07-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录