go面向接口编程知识点

接口定义与格式

接口(interface)是一种类型,用来定义行为(方法)。这句话有两个重点,类型定义行为

首先解释定义行为:
接口即一组方法定义的集合,定义了对象的一组行为,就是定义了一些函数,由具体的类型实例实现具体的方法。
换句话说,一个接口就是定义(规范或约束),接口并不会实现这些方法,具体的实现由类实现,实现接口的类必须严格按照接口的声明来实现接口提供的所有功能。接口的作用应该是将定义与实现分离,降低耦合度。
在多人合作开发同一个项目时,​接口表示调用者和设计者的一种约定,事先定义好相互调用的接口可以大大提高开发的效率。有了接口,就可以在不影响现有接口声明的情况下,修改接口的内部实现,从而使兼容性问题最小化。

接口的定义格式:

type Namer interface {
Method1(param_list) return_type //方法名(参数列表) 返回值列表
Method2(param_list) return_type //方法名(参数列表) 返回值列表
   ......
 }

  

隐式实现及实现条件

怎么实现接口:
实现接口的类并不需要显式声明,只需要实现接口所有的函数就表示实现了该接口,而且类还可以拥有自己的方法。

接口能被哪些类型实现:
接口可以被结构体实现,也可以被函数类型实现。
接口被实现的条件:
接口被实现的条件一:接口的方法与实现接口的类型方法格式一致(方法名、参数类型、返回值类型一致)。
接口被实现的条件二:接口中所有方法均被实现。

package main

import "fmt"

type Shaper interface {
Area() float64
// Perimeter() float64
} type Rectangle struct {
length float64
width float64
} // 实现 Shaper 接口中的方法
func (r *Rectangle) Area() float64 {
return r.length * r.width
} // Set 是属于 Rectangle 自己的方法
func (r *Rectangle) Set(l float64, w float64) {
r.length = l
r.width = w
} func main() {
rect := new(Rectangle) //创建指针类型的结构体实例(类实例)
rect.Set(2, 3)
areaIntf := Shaper(rect) //这里将指针类型实例赋值给接口,下面会介绍。
fmt.Printf("The rect has area: %f\n", areaIntf.Area())
}

  

接口赋值

现在来解释接口是一个类型,本质是一个指针类型,那么什么样的值可以赋值给接口,有两种:实现了该接口的类或者接口

1.将对象赋值给接口
当接口实例中保存了自定义类型的实例后,就可以直接从接口上调用它所保存的实例的方法。

package main

import (
"fmt"
) //定义接口
type Testinterface interface{
Teststring() string
Testint() int
} //定义结构体
type TestMethod struct{
name string
age int
} //结构体的两个方法隐式实现接口
func (t *TestMethod)Teststring() string{
return t.name
} func (t *TestMethod)Testint() int{
return t.age
} func main(){
T1 := &TestMethod{"ling",34}
T2 := TestMethod{"gos",43}
//接口本质是一种类型
//接口赋值:只要类实现了该接口的所有方法,即可将该类赋值给这个接口
var Test1 Testinterface //接口只能是值类型
Test1 = T1 //TestMethod类的指针类型实例传给接口
fmt.Println(Test1.Teststring())
fmt.Println(Test1.Testint()) Test2 := T2 //TestMethod类的值类型实例传给接口
fmt.Println(Test2.Teststring())
fmt.Println(Test2.Testint())
}

2.将接口赋值给另一个接口

1.只要两个接口拥有相同的方法列表(与次序无关),即是两个相同的接口,可以相互赋值
2.接口赋值只需要接口A的方法列表是接口B的子集(即假设接口A中定义的所有方法,都在接口B中有定义),那么B接口的实例可以赋值给A的对象。反之不成立,即子接口B包含了父接口A,因此可以将子接口的实例赋值给父接口。
3.即子接口实例实现了子接口的所有方法,而父接口的方法列表是子接口的子集,则子接口实例自然实现了父接口的所有方法,因此可以将子接口实例赋值给父接口。

 3.接口类型作为参数

第一点已经说了可以将实现接口的类赋值给接口,而将接口类型作为参数很常见。这时,那些实现接口的实例都能作为接口类型参数传递给函数/方法。

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
//将 Circle的指针类型实例c1传给函数myArea,接收类型为Shaper接口
myArea(c1)
}
func myArea(n Shaper) {
fmt.Println(n.Area())
}

  

空接口

空接口是指没有定义任何接口方法的接口。没有定义任何接口方法,意味着Go中的任意对象都已经实现空接口(因为没方法需要实现),只要实现接口的对象都可以被接口保存,所以任意对象都可以保存到空接口实例变量中

空接口的定义方式:

type empty_int interface {}

更常见的,会直接使用interface{}作为一种类型,表示空接口。例如:

// 声明一个空接口实例
var i interface{}

再比如函数使用空接口类型参数:

func myfunc(i interface{})

如何使用空接口

可以定义一个空接口类型的array、slice、map、struct等,这样它们就可以用来存放任意类型的对象,因为任意类型都实现了空接口。

package main

import "fmt"

func main() {
any := make([]interface{}, 5)
any[0] = 11
any[1] = "hello world"
any[2] = []int{11, 22, 33, 44}
for _, value := range any {
fmt.Println(value)
}
}
11
hello world
[11 22 33 44]
<nil>
<nil>

通过空接口类型,Go也能像其它动态语言一样,在数据结构中存储任意类型的数据。

 

接口嵌套

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

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

type ReadWrite interface {
Read(b Buffer) bool
Write(b Buffer) bool
} type Lock interface {
Lock()
Unlock()
} type File interface {
  //ReadWrite为内部接口
ReadWrite
  //Lock为内部接口
Lock
Close()
}

  

类型断言

类型断言为判断一个类型有没有实现接口。

假如我现在写了一个结构体类型 MyFile 来实现上面的 File 接口,那么我如何知道 MyFile 是否实现了 File 接口呢?

package main

import "fmt"

type MyFile struct{}

func (m *MyFile) Read() bool {
fmt.Printf("Read()\n")
return true
} // ...
// 假设我这里相继实现了 Write(), Lock(),Unlock() 和 Close() 方法 func main() {
my := new(MyFile)
fIntf := File(my) // 看这里,看这里
if v, ok := fIntf.(*MyFile); ok {
v.Read()
}
}  

类型断言的格式:

if v, ok : = varI.(T) ; ok { 
   // checked type assertion
//do something
return
}

如果 v 是 varI 转换到类型 T 的值,ok 会是 true;否则 v 是类型 T 的零值,ok 是 false。 

要是多个类型实现了同一个接口,比如前面的 areaIntf,要如何测试呢? 
那就要用 type-switch 来判断了。

switch t := areaIntf.(type) {
case *Rectangle:
// do something
case *Triangle:
// do something
default:
// do something
}

  

多态

1、多个类型(结构体)可以实现同一个接口。
2、一个类型(结构体)可以实现多个接口。
3、实现接口的类(结构体)可以赋值给接口。

package main

import "fmt"

type Shaper interface {
Area() float64
} // ==== Rectangle ====
type Rectangle struct {
length float64
width float64
} // 实现 Shaper 接口中的方法
func (r *Rectangle) Area() float64 {
return r.length * r.width
} // Set 是属于 Rectangle 自己的方法
func (r *Rectangle) Set(l float64, w float64) {
r.length = l
r.width = w
} // ==== Triangle ====
type Triangle struct {
bottom float64
hight float64
} func (t *Triangle) Area() float64 {
return t.bottom * t.hight / 2
} func (t *Triangle) Set(b float64, h float64) {
t.bottom = b
t.hight = h
} // ==== Triangle End ==== func main() {
rect := new(Rectangle)
rect.Set(2, 3)
areaIntf := Shaper(rect) //这种方法只能将指针类型的类示例赋值给接口
fmt.Printf("The rect has area: %f\n", areaIntf.Area()) triangle := new(Triangle)
triangle.Set(2, 3)
areaIntf = Shaper(triangle) //这种方法只能将指针类型的类示例赋值给接口
fmt.Printf("The triangle has area: %f\n", areaIntf.Area())
}

 

go接口详解的更多相关文章

  1. JDBC常用接口详解

    JDBC中常用接口详解 ***DriverManager 第一.注册驱动 第一种方式:DriverManager.registerDriver(new com.mysql.jdbc.Driver()) ...

  2. Java6.0中Comparable接口与Comparator接口详解

    Java6.0中Comparable接口与Comparator接口详解 说到现在,读者应该对Comparable接口有了大概的了解,但是为什么又要有一个Comparator接口呢?难道Java的开发者 ...

  3. socket接口详解

    1. socket概述 socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用已实现进程在网络中通信. socket起源于UNIX,在Unix一切 ...

  4. “全栈2019”Java第八十四章:接口中嵌套接口详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  5. “全栈2019”Java第八十三章:内部类与接口详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  6. Java接口 详解(二)

    上一篇Java接口 详解(一)讲到了接口的基本概念.接口的使用和接口的实际应用(标准定义).我们接着来讲. 一.接口的应用—工厂设计模式(Factory) 我们先看一个范例: package com. ...

  7. [转载]MII/MDIO接口详解

    原文地址:MII/MDIO接口详解作者:心田麦浪 本文主要分析MII/RMII/SMII,以及GMII/RGMII/SGMII接口的信号定义,及相关知识,同时本文也对RJ-45接口进行了总结,分析了在 ...

  8. map接口详解

    1.Map接口详解(1)映射(map)是一个存储键.键值对的对象,给定一个键,可以查询得到它的值,键和值都可以是对象(2)键必须是唯一的,值可以重复(Map接口映射唯一的键到值)(3)有些映射可以接收 ...

  9. ReadWriteLock 接口详解

    ReadWriteLock 接口详解 这是本人阅读ReadWriteLock接口源码的注释后,写出的一篇知识分享博客 读写锁的成分是什么? 读锁 Lock readLock(); 只要没有写锁,读锁可 ...

  10. java.io.DataInput接口和java.io.DataOutput接口详解

    public interface DataInput DataInput 接口用于从二进制流中读取字节,并重构所有 Java 基本类型数据.同时还提供根据 UTF-8 修改版格式的数据重构 Strin ...

随机推荐

  1. C#录制视频

    这是一个使用C#语言制作的录制框架,支持录制桌面,多屏,声音,摄像头,某个应用程序的界面 1.安装 使用此框架需要安装扩展包Kogel.Record,可以Nuget上搜索 或者使用Nuget命令 In ...

  2. 微信小程序之双重循环(包含左滑删除,以及数据各项处理)

    <view wx:for="{{hommer}}" wx:for-item="item" wx:for-index="index" w ...

  3. C++string中find,find_first_of和find_last_of的用法

    1. size_t find (const string& str, size_t pos = 0) str.find(str1) 说明:从pos(默认是是0,即从头开始查找)开始查找,找到第 ...

  4. Redis系列之----Redis的两种持久化机制(RDB和AOF)

    Redis的两种持久化机制(RDB和AOF) 什么是持久化    Redis的数据是存储在内存中的,内存中的数据随着服务器的重启或者宕机便会不复存在,在生产环境,服务器宕机更是屡见不鲜,所以,我们希望 ...

  5. 网络连接报错“DNS服务未响应”

    一般报这个错误就是网络没有正常连接. 先检查连接情况:路由器是否正常.网线是否正常.接口是否正常.

  6. 使用“1”个参数调用“DownloadString”时发生异常:“操作超时”

    我今天在终端美化时间遇到一个问题是这样的 使用“1”个参数调用“DownloadString”时发生异常:“操作超时” 然后网我看了下,访问链接属于https的东西,根据直觉我觉得是这样的,是由于访问 ...

  7. 逆元(inv)

    推荐博客 : http://blog.csdn.net/baidu_35643793/article/details/75268911 通常我们在计算除法取模时,并不能直接的取模后再去相除,答案会有问 ...

  8. 2019 牛客国庆集训day1 2019 点分治

    题目链接:https://ac.nowcoder.com/acm/contest/1099/I 点分治,计算路径数的时候,先将每个点到根的距离模2019,计算的时候就可以O(n)求出数目,对于模201 ...

  9. 基于iTextSharp的PDF操作(PDF打印,PDF下载)

    基于iTextSharp的PDF操作(PDF打印,PDF下载) 准备 1. iTextSharp的简介 iTextSharp是一个移植于java平台的iText项目,被封装成c#的组件来用于C#生成P ...

  10. 【python系统学习07】一张图看懂字典并学会操作

    点击跳转 - 原文地址 数据类型 - 字典(dict) 目录: 一张图get字典 字典是什么 js的对象 字典长啥样 语法伪代码 示例demo 语法成像 字典怎么用 字典长度获取--len函数 提取字 ...