从零开始学Golang的接口

目录
  • 前言
  • 1.为什么需要接口?
  • 2.接口是什么?如何定义?
  • 3.接口实战初体验
  • 4.如何测试是否已实现该接口?
  • 5.空接口&类型断言
  • 6.接口零值
  • 7.一个类型实现多个接口
  • 8.指针与值类型实现接口的区别
  • 9.接口嵌套

前言

接口在面向对象编程中是经常使用的招式,也是体现多态很重要的手段。
是的。Golang中也有接口这玩意儿。

1.为什么需要接口?

多数情况下,数据可能包含不同的类型,却会有一个或者多个共同点,这些共同点就是抽象的基础。前文讲到的Golang继承解决的是is-a的问题,单一继承的关系。但是当不同的父类具有相同的行为的时候,单一继承就没法解决了。

于是乎,接口出现了。接口可以理解为某一个方面的抽象,可以是多对一的(多个类型实现一个接口),这也是多态的体现。解决了上文一对一的问题。

2.接口是什么?如何定义?

是什么
接口是一组仅包含方法名、参数、返回值的未具体实现的方法的集合。

如果实现了接口的所有方法,则认为实现了该接口,无需在该类型上显示的添加声明。

这个解释下,加深印象,在php中接口是长这样的:

//定义接口
interface base{
   public function getName();
}
 
//学生类
class student implements base{
   public function getName(){
      echo "咖啡色的羊驼";
   }
}

这里有个关键字:implements。

这样的声明称之为显示的,而在Golang中接口是隐式地实现。(埋个伏笔看下文)

定义

type interfaceName interface { 
    // 方法列表 
    GetName() string
} 

3.接口实战初体验

实际编程中呢,接口的命名大伙儿喜欢使用er结尾。当然这个看个人喜好。

上代码:

    package main

    import (
        "fmt"
    )
    
    // 定义一个接口
    type People interface {
        ReturnName() string
    }
    
    // 定义一个结构体
    type Student struct {
        Name string
    }
    
    // 定义结构体的一个方法。
    // 突然发现这个方法同接口People的所有方法(就一个),此时可直接认为结构体Student实现了接口People
    func (s Student) ReturnName() string {
        return s.Name
    }
    
    func main() {
        cbs := Student{Name:"咖啡色的羊驼"}
    
        var a People
        // 因为Students实现了接口所以直接赋值没问题
        // 如果没实现会报错:cannot use cbs (type Student) as type People in assignment:Student does not implement People (missing ReturnName method)
        a = cbs       
        name := a.ReturnName() 
        fmt.Println(name) // 输出"咖啡色的羊驼"
    }

4.如何测试是否已实现该接口?

使用接口特有的断言判断来实现(下文还会再次提到,加深印象)。

语法:x.(T)
这样的语法只适应于x是interface类型

接着上文例子,继续上代码:

    // 由于x.(T)只能是接口类型判断,所以传参时候,传入的是接口类型
    // 为何test的类型可以是一个空接口?埋伏笔下文便知。
    func CheckPeople(test interface{}) {
        if _, ok := test.(People); ok {
            fmt.Printf("Student implements People")
        }
    }

    
    func main() {
        cbs := Student{Name:"咖啡色的羊驼"}
        CheckPeople(cbs) // Student implements People
    }

5.空接口&类型断言

空接口

空接口就是不包含任何方法的接口。正因为如此,所有的类型都实现了空接口。

虽然空接口起不到任何作用,但是空接口在需要存储任何类型数值的时候非常有用,这也回答了上文的问题,因为空接口可以存储任意类型的数据。

    // 定义cbs为空接口
    var cbs interface{}
    var i int = 5
    var s string = "Hello world"
    // cbs可以存储任意类型的数值
    cbs = i
    cbs = s

类型断言

既然空接口可以存储任意类型,那么如何区分不同的类型?
常用的有两种方法:Comma-ok断言、switch判断。

上代码:

    package main
    
    import (
        "fmt"
    )
    
    // 定义一个结构体
    type Student struct {
        Name string
    }
    
    // 类型断言
    func main() {
        Params := make([]interface{}, 3)
        Params[0] = 88                   // 整型
        Params[1] = "咖啡色的羊驼"         // 字符串
        Params[2] = Student{Name: "cbs"} // 自定义结构体类型
        
        // Comma-ok断言
        for index, v := range Params {
            if _, ok := v.(int); ok {
                fmt.Printf("Params[%d] 是int类型 \n", index)
            } else if _, ok := v.(string); ok {
                fmt.Printf("Params[%d] 是字符串类型\n", index)
            } else if _, ok := v.(Student); ok {
                fmt.Printf("Params[%d] 是自定义结构体类型\n", index)
            } else {
                fmt.Printf("list[%d] 未知类型\n", index)
            }
        }
        
        // switch判断
        for index, v := range Params {
            switch  value := v.(type) {
            case int:
                fmt.Printf("Params[%d] 是int类型, 值:%d \n", index,value)
            case string:
                fmt.Printf("Params[%d] 是字符串类型, 值:%s\n", index,value)
            case Student:
                fmt.Printf("Params[%d] 是Person类型, 值:%s\n", index,value)
            default:
                fmt.Printf("list[%d] 未知类型\n", index)
            } 
        
        }  
    }

6.接口零值

接口的零值是nil

    package main

    import (
        "fmt"
    )

    type People interface {
        GetName() string
    }

    // 输出 "cbs is nil 类型"
    func main() {
        var cbs People
        if cbs == nil {
            fmt.Println("cbs is nil 类型")
        }
    }

7.一个类型实现多个接口

    package main

    import (
        "fmt"
    )

    type People interface {
        ReturnName() string
    }

    type Role interface {
        ReturnRole() string
    }

    type Student struct {
        Name string
    }

    func (s Student) ReturnName() string {
        return s.Name
    }

    func (s Student) ReturnRole() string {
        return "学生"
    }

    func main() {
        cbs := Student{Name: "咖啡色的羊驼"}

        var a People  // 定义a为People接口类型
        var b Role    // 定义b为Role接口类型

        a = cbs // 由于Student实现了People所有方法,所以接口实现成功,可直接赋值
        b = cbs // 由于Student实现了Role所有方法,所以接口实现成功,可直接赋值

        name := a.ReturnName()
        fmt.Println(name) // 输出"咖啡色的羊驼"

        role := b.ReturnRole()
        fmt.Println(role) // 输出"学生"
    }

也说明一个东西:实现了某个接口的类型,还可以有其它的方法。只要是方法实现包含接口的即可。

8.指针与值类型实现接口的区别

    package main

    import (
        "fmt"
    )

    type People interface {
        ReturnName() string
    }

    type Student struct {
        Name string
    }

    type Teacher struct {
        Name string
    }

    func (s Student) ReturnName() string {
        return s.Name
    }

    func (t *Teacher) ReturnName() string {
        return t.Name
    }

    func main() {
        cbs := Student{Name: "咖啡色的羊驼"}
        sss := Teacher{Name: "咖啡色的羊驼的老师"}

        // 值类型
        var a People
        a = cbs
        name := a.ReturnName()
        fmt.Println(name)

        // 指针类型
        // a = sss <- 这样写不行!!!
        a = &sss // 由于是指针类型,所以赋值的时候需要加上&
        name = a.ReturnName()
        fmt.Println(name) // 输出"咖啡色的羊驼的老师"
    }

"a = sss"这样写会发生报错:

cannot use sss (type Teacher) as type People in assignment:
    Teacher does not implement People (ReturnName method has pointer receiver)

因为是Teacher的指针实现了ReturnName方法,Teacher本身没实现。

9.接口嵌套

类似于PHP的接口继承,Golang也有它的接口嵌套。

    package main

    import (
        "fmt"
    )

    type People interface {
        ReturnName() string
    }

    type Role interface {
        People // 接口嵌套
        ReturnRole() string
    }

    type Student struct {
        Name string
    }

    func (s Student) ReturnName() string {
        return s.Name
    }

    func (s Student) ReturnRole() string {
        return "学生"
    }

    func main() {
        cbs := Student{Name: "咖啡色的羊驼"}

        var a Role
        a = cbs
        name := a.ReturnName()
        fmt.Println(name)

        role := a.ReturnRole()
        fmt.Println(role)
    }

到此这篇关于从零开始学Golang的接口的文章就介绍到这了,更多相关Golang 接口内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 初步解读Golang中的接口相关编写方法

    概述 如果说goroutine和channel是Go并发的两大基石,那么接口是Go语言编程中数据类型的关键.在Go语言的实际编程中,几乎所有的数据结构都围绕接口展开,接口是Go语言中所有数据结构的核心. Go语言中的接口是一些方法的集合(method set),它指定了对象的行为:如果它(任何数据类型)可以做这些事情,那么它就可以在这里使用. 接口的定义和使用 比如 复制代码 代码如下: type I interface{     Get() int     Put(int)   } 这段话就定

  • golang接口IP限流,IP黑名单,IP白名单的实例

    增加中间件 可以选择普通模式和LUA脚本模式,建议选择普通模式,实际上不需要控制的那么精确. package Middlewares import ( "github.com/gin-gonic/gin" "strconv" "time" "voteapi/pkg/app/response" "voteapi/pkg/gredis" "voteapi/pkg/util" ) const

  • golang中接口对象的转型两种方式

    接口对象的转型有两种方式: 1. 方式一:instance,ok:=接口对象.(实际类型) 如果该接口对象是对应的实际类型,那么instance就是转型之后对象,ok的值为true 配合if...else if...使用 2. 方式二: 接口对象.(type) 配合switch...case语句使用 示例: package main import ( "fmt" "math" ) type shape interface { perimeter() int area

  • golang基础之Interface接口的使用

    接口是一个或多个方法签名名的集合,定义方式如下 type Interface_Name interface { method_a() string method_b() int .... } 只要某个类型拥有该接口的所有方法签名,就算实现该接口,无需显示声明实现了那个接口,这称为structural Typing package main import "fmt" type USB interface { //定义一个接口:方法的集合 Name() string //Name方法,返回

  • 如何判断Golang接口是否实现的操作

    前言 在看一个底层库的的时候,看到了一个比较奇怪的写法,于是乎有了本文. 主要探讨两个问题: 1.利用编译来判断Golang接口是否实现 2.延伸出的make和new的区别 正文 1.利用编译来判断Golang接口是否实现 看了一个底层通用链接池的库,有这么一行代码: var _ Pooler = new(WeightedRoundRobin) 需要解释的是:Pooler是一个接口类型. type Pooler interface { // ... } 刚开始看是疑惑的,为什么new了之后是要抛

  • golang 接口嵌套实现复用的操作

    大家还是直接看代码吧~ package main import ( "fmt" ) func main() { start(NewB(C{})) start(NewB(D{})) } type A interface { what() } type B struct { A } type C struct { } func (b C) what() { fmt.Println("this is type C") } type D struct { } func (b

  • Golang 使用接口实现泛型的方法示例

    在C/C++中我们可以使用泛型的方法使代码得以重复使用,最常见例如stl functions:vector<int> vint or vector<float> vfloat等.这篇文章将使用interface{...}接口使Golang实现泛型. interface{...}是实现泛型的基础.如一个数组元素类型是interface{...}的话,那么实现了该接口的实体都可以被放置入数组中.注意其中并不一定必须是空接口(简单类型我们可以通过把他转化为自定义类型后实现接口).为什么i

  • golang中interface接口的深度解析

    一 接口介绍 如果说gorountine和channel是支撑起Go语言的并发模型的基石,让Go语言在如今集群化与多核化的时代成为一道亮丽的风景,那么接口是Go语言整个类型系列的基石,让Go语言在基础编程哲学的探索上达到前所未有的高度.Go语言在编程哲学上是变革派,而不是改良派.这不是因为Go语言有gorountine和channel,而更重要的是因为Go语言的类型系统,更是因为Go语言的接口.Go语言的编程哲学因为有接口而趋于完美.C++,Java 使用"侵入式"接口,主要表现在实现

  • 从零开始学Golang的接口

    目录 前言 1.为什么需要接口? 2.接口是什么?如何定义? 3.接口实战初体验 4.如何测试是否已实现该接口? 5.空接口&类型断言 6.接口零值 7.一个类型实现多个接口 8.指针与值类型实现接口的区别 9.接口嵌套 前言 接口在面向对象编程中是经常使用的招式,也是体现多态很重要的手段.是的.Golang中也有接口这玩意儿. 1.为什么需要接口? 多数情况下,数据可能包含不同的类型,却会有一个或者多个共同点,这些共同点就是抽象的基础.前文讲到的Golang继承解决的是is-a的问题,单一继承

  • Spring Boot集成Redis实现缓存机制(从零开始学Spring Boot)

    本文章牵涉到的技术点比较多:spring Data JPA.Redis.Spring MVC,Spirng Cache,所以在看这篇文章的时候,需要对以上这些技术点有一定的了解或者也可以先看看这篇文章,针对文章中实际的技术点在进一步了解(注意,您需要自己下载Redis Server到您的本地,所以确保您本地的Redis可用,这里还使用了MySQL数据库,当然你也可以内存数据库进行测试).这篇文章会提供对应的Eclipse代码示例,具体大体的分如下几个步骤: (1)新建Java Maven Pro

  • 从零开始学springboot整合feign跨服务调用的方法

    介绍 微服务横行的互联网世界, 跨服务调用显得很平凡, 我们除了采用传统的http方式接口调用, 有没有更为优雅方便的方法呢? 答案是肯定的,feign就提供了轻便的方式! 如果你的服务都注册了注册中心,比如nacos, 那么调用会显得很轻松, 只需一个注解, 带上需要调用的服务名即可,**feign + nacos**会帮你做剩余的事. 如果没有注册中心, 也无需担心, feign一样可以以传统的 ip:port 方式进行调用~ 下面,我们来实践下吧 springboot整合feign 引入依

  • 从零开始学SpringBoot如何开始使用图文详解

    目的: <从零开始学SpringBoot>,是小编打算通过写一系列的文章,让大家能够认识SpringBoot,通过对SpringBoot的入门学习后,小编会在通过一个示例Demo来让大家能够真正上手SpringBoot. 适合人群: 1.有一定Java基础的朋友 2.适合初中级的朋友. 如果文章编写中存在问题或者对文章有疑问,都可以留言小编,和小编一起探讨,小编会虚心接受大家的建议并更正. 1.什么是Spring Boot 来源官方文档: Spring Boot makes it easy t

  • 深入Golang的接口interface

    目录 前言 接口转换的原理 实现多态 前言 go不要求类型显示地声明实现了哪个接口,只要实现了相关的方法即可,编译器就能检测到 空接口类型可以接收任意类型的数据: type eface struct { // _type 指向接口的动态类型元数据 // 描述了实体类型.包括内存对齐方式.大小等 _type *_type // data 指向接口的动态值 data unsafe.Pointer } 空接口在赋值时,_type 和 data 都是nil.赋值后,_type 会指向赋值的数据元类型,d

  • 从零开始学ASP.NET-基础篇第1/7页

    第一天 学习目的: 掌握最基本的Label.TextBox.Button控件用法 掌握用StringBuider类连接字符串 理解服务器的环境变量 StringBuilder类: 命名空间是:System.Text. StringBuilder类是个高效的类,StringBuilder.Append连接字符串的方法是非常快的.用于连接大量的字符串,其速度的优越性就会体现出来. 先举几个例子: 在cs或vb文件的头部加上 [C#]using System.Text; [VB]Imports Sys

  • 从零开始学JAVA之可变参数

    下面是一个简单的小程序: 复制代码 代码如下: import java.util.Arrays; class lesson6 {     public static void main(String[] args)     {         int array[]={2,3,1,5,4,6};         System.out.println(Arrays.toString(array));         System.out.println(getSum(array));       

  • 从零开始学YII2框架(二)通过 Composer 安装扩展插件

    目前yii2的扩展还不是很多,截止到今天,在官网一共有33个,不过这些插件中不乏有优秀的扩展插件, 我尝试了几个,发现了一系列好用的Yii2插件,作者是来自印度的krajee团队,他们写的插件都很好用.推荐一下. krajee团队的网站:http://krajee.com,有几个不错的插件可以尝试. 下面来介绍Yii2的插件安装方法.通过Composer安装插件yii2-detail-view. Git 推荐安装Git,Composer安装插件时候会用到Git Clone,Git官方下载网站:传

随机推荐