一:接口概要


接口是一种重要的类型,他是一组确定的方法集合。

一个接口变量可以存储任何实现了接口方法的具体值。
一个重要的例子就是io.Reader和io.Writer

type Reader interface {
Read(p []byte) (n int, err error)
} type Writer interface {
Write(p []byte)(n int, err error)
}

如果一个方法实现了io.Reader或者io.Writer里的方法,那么它便实现了io.Reader 或 io.Writer 。 这就

意味着一个io.Reader 变量可以持有任何一个实现了Read方法的类型的值 [1]

例 1:
var r io.Reader
r = os.Stdin
os.Stdin 在 os 包中定义:

var (
Stdin = NewFile(uintptr(syscall.Stdin), "/dev/stdin")
Stdout = NewFile(uintptr(syscall.Stdout), "/dev/stdout")
)

NewFile 返回的是一个 *File 结构体,在 os/types.go, File 结构体代码,里面是一个匿名的file结构体

// File represents an open file descriptor.
type File struct {
*file // os specific
}

file也是一个结构体,os/file_plan9.go

type file struct {
fd int
name string
dirinfo *dirInfo // nil unless directory being read
}

这个 File 结构体实现了 Read(p []byte) (n int, err error) 方法, 而这个刚好是接口 io.Reader 里的方法
所以os.Stdin 可以赋值给 io.Reader

例 2:
var r io.Reader
r = bufio.NewReader(r)

在包 bufio.go

// NewReader returns a new Reader whose buffer has the default size.
func NewReader(rd io.Reader) *Reader {
return NewReaderSize(rd, defaultBufSize)
}
func NewReaderSize(rd io.Reader, size int) *Reader {
// Is it already a Reader?
b, ok := rd.(*Reader)
if ok && len(b.buf) >= size {
return b
}
if size < minReadBufferSize {
size = minReadBufferSize
}
r := new(Reader)
r.reset(make([]byte, size), rd)
return r
}

2)上面[1]中下划线 接口变量可以持有任何一个实现了Read方法的类型的值,那么接口变量又是什么呢?
变量里是具体值和这个值类型

3)一个类型实现了 interface 中的所有方法,我们就说该类型实现了该 interface,
所以所有的类型都实现了 empty interface,因为任何类型至少有0个方法。
go中没有显示的关键字来实现interface,只需要实现interface包含的方法即可

二: 接口定义


interface类型定义

type interfaceNamer interface {
Method1(param_list) return_type
Method2(param_list) return_type
...
}

按照约定,只包含一个方法的)接口的名字由方法名加 [e]r 后缀组成,例如 Printer、Reader、Writer、Logger、Converter 等等。还有一些不常用的方式(当后缀 er 不合适时),比如 Recoverable,此时接口名以 able 结尾,或者以 I 开头(像 .NET 或 Java 中那样)

接口可以屏蔽内部细节,和具体的实现方法。只关注我需要做什么,而不关注怎么做

三:接口和方法,实现接口的方法


示例:

package main

import (
"fmt"
) type Ager interface {
Get() int
Set(int)
} type Student struct {
Age int
} func (s Student) Get() int {
return s.Age
} func (s *Student) Set(age int) {
s.Age = age
} func funone(ager Ager) {
ager.Set()
fmt.Println(ager.Get())
} func main() {
//=== 第一部分 stu 可以作为一个变量传递给函数,因为它作为一个struct实现了接口里面的方法
stu := Student{}
funone(&stu)
//=== 第二部分 i 作为一个接口变量,它存储了实现了这个接口方法的类型的值
var i Ager
i = &stu
fmt.Println(i.Get())
}

在上面程序中,定义来一个名为Ager的 interface 接口,里面定义了2个方法Get() , Set(), 定义了一个 student 的结构体,并且结构体实现了 interface 里面的2个方法Get,Set,我们就说结构体实现了 这个接口
定义了一个 funcone 的函数,里面参数ager是一个interface类型(interface类型作为函数参数),那么实现了这个接口的任何变量都可以作为参数传递给 funcone 了, (在 main 函数中 第一部分注释)
interface变量存储了实现者变量的值 (main函数中的 第二部分注释),所以它又可以调用结构体的方法

一个类型实现了接口里的所有方法,那么这个类型就实现了这个接口

四:接口嵌套


一个接口可以包含一个或者多个其他接口

相当于直接将这些内嵌接口的方法加入到了外层接口中,相当于组合了方法

还是拿最典型的 Reader 和 Writer 接口

type Reader interface {
Read(p []byte) (n int, err error)
} type Writer interface {
Write(p []byte) (n int, err error)
} type ReadWriter interface {
Reader
Writer
}

如果一个类型既实现了 Reader 接口,也实现了Writer接口,那么它就自动实现了 ReadWriter 接口
接口和结构体的定义很相似,也可以完成嵌入接口的功能,嵌入的匿名的接口,可以自动的具备被嵌入的接口的方法

五: 空interface


interface{} 空interface,它能存储任意类型的值, 它相当于 c 系的 void * 类型

定义一个空接口

var i interface{}
s := "hello"
num :
i = s
i = num

空接口在打印包中的应用,它又是一个可变长参数 fmt/print.go

func Println(a ...interface{}) (n int, err error) {
return Fprintln(os.Stdout, a...)
}

六:类型断言


既然接口类型能够实现多种类型的值,那么我们应该有一种方式来检测这种动态类型的值,即运行时变量中存储的值的实际类型。 在执行过程中动态类型可能会有所不同,但是它总是可以分配给接口变量本身的类型。

那就是 类型断言
1):type-ok 形式
1.1:第一种
直接判断是否是某类型的变量 value,ok = element.(T), 这里value是变量的值,ok是bool类型, element必须是一个 interface 变量, T是断言的类型

1.2 第二种 简单形式:
value := element.(T)

示例:

package main

import (
"fmt"
"math"
) type Square struct {
side float32
} type Circle struct {
radius float32
} type Shaper interface {
Area() float32
} func main() {
var area Shaper
sq1 := new(Square)
sq1.side = area = sq1
//is Square the type of area
if t, ok := area.(*Square); ok {
fmt.Printf("The type of area is: %T \n", t)
} if u, ok := area.(*Circle); ok {
fmt.Printf("The type of area is: %T \n", u)
} else {
fmt.Println("area does not cotain a variable of type Cicle")
}
//输出结果
// go run assert1.go
// The type of area is: *main.Square
// area does not cotain a variable of type Cicle } func (sq *Square) Area() float32 {
return sq.side * sq.side
} func (ci *Circle) Area() float32 {
return ci.radius * ci.radius * math.Pi
}

2):switch 形式
上面也可以用 switch的形式

switch t := area.(type) {
case *Square:
fmt.Printf("Type Square %T with value %v\n", t, t)
case *Circle:
fmt.Printf("Type Circle %T with value %v\n", t, t)
case nil:
fmt.Printf("nil value: nothing to check?\n")
default:
fmt.Printf("Unexpected type %T\n", t)
}

参考:
1. http://wiki.jikexueyuan.com/project/the-way-to-go/11.4.html
2. https://research.swtch.com/interfaces
3. https://blog.golang.org/laws-of-reflection

go interface接口的更多相关文章

  1. as3.0 interface接口使用方法

    [转]as3.0 interface接口使用方法 AS在2.0的时候就支持接口了 接口能够让你的程序更具扩展性和灵活性,打个例如 比方你定义了一个方法 代码: public function aMet ...

  2. interface接口

    当一个抽象类中的方法都是抽象的时候,这时可以将该抽象类用另一种形式定义和表示,就是接口 interface. 定义接口使用的关键字不是class,是interface.接口中常见的成员: 这些成员都有 ...

  3. golang面向对象和interface接口

    一. golang面向对象介绍 1.golang也支持面向对象编程,但是和传统的面向对象编程有区别,并不是纯粹的面向对象语言.2.golang没有类(class),golang语言的结合体(struc ...

  4. Golang 入门系列(四)如何理解interface接口

    前面讲了很多Go 语言的基础知识,包括go环境的安装,go语言的语法等,感兴趣的朋友,可以先看看之前的文章.https://www.cnblogs.com/zhangweizhong/category ...

  5. java interface接口的传值方法

    A 类 package interface_test; public class A { private IPresenter ip; public A(IPresenter ip) { this.i ...

  6. JAVA 构造器, extends[继承], implements[实现], Interface[接口], reflect[反射], clone[克隆], final, static, abstrac

    记录一下: 构造器[构造函数]: 在java中如果用户编写类的时候没有提供构造函数,那么编译器会自动提供一个默认构造函数.它会把所有的实例字段设置为默认值:所有的数字变量初始化为0;所有的布尔变量设置 ...

  7. 011-对象——interface接口说明与使用方式实例

    <?php /** interface接口说明与使用方式实例 * * 接口里面的方法全是抽象方法,没有实体的方法.这样的类我们就叫做接口.定义的时候用Interface定义.实现接口时用impl ...

  8. Java Interface接口

    Java 中接口概念 接口可以理解为一种特殊的 类,由 全局常量 和 公共的抽象方法 所组成. 类是一种具体实现体,而接口定义了某一批类所需要遵循的规范,接口不关心这些类的内部数据, 也不关心这些类里 ...

  9. Golang基础(8):go interface接口

    一:接口概要 接口是一种重要的类型,他是一组确定的方法集合. 一个接口变量可以存储任何实现了接口方法的具体值.一个重要的例子就是io.Reader和io.Writer type Reader inte ...

随机推荐

  1. python 通过元类控制类的创建

    一.python中如何创建类? 1. 直接定义类 class A: a = 'a' 2. 通过type对象创建 在python中一切都是对象 在上面这张图中,A是我们平常在python中写的类,它可以 ...

  2. WinServer-FTP搭建

    FTP服务器(File Transfer Protocol Server)是在互联网上提供文件存储和访问服务的计算机,它们依照FTP协议提供服务. FTP是File Transfer Protocol ...

  3. MPP架构海量数据分析仓库——Greenplum介绍

    一.Greenplum背景 时间回到2002年,互联网行业经过近10年的发展,数据量正处于快速增长期: 1.传统的主机计算模式在海量数据面前,除了造价昂贵外,在CPU计算和IO吞吐上不能满足海量数据的 ...

  4. LSB和MSB

    最低有效位(the least significant bit,lsb)是指一个二进制数字中的第0位(即最低位),具有权值为2^0,可以用它来检测数的奇偶性.与之相反的称之为最高有效位.在大端序中,l ...

  5. navicat 将自增长字段重置(重新从1开始)的方法

    先说明,此语句会将你的表中数据全部删除. 很简单,运行如下sql语句: TRUNCATE TABLE 表名;

  6. Linux shell 及命令汇总

    1 文件管理命令 1.cat命令:将文件内容连接后传送到标准输出或重定向到文件 2.chmod命令:更改文件的访问权限 3.chown命令:更改文件的所有者 4.find命令:查找(符合条件)文件并将 ...

  7. 浪潮服务器I4008/NX5480M4介绍

    浪潮I4008 / NX5480M4是一款高密度模块化服务器. I4008是机箱,NX5480M4是节点. 8个计算节点模块可以部署在标准机架4U高度机器里,具有高性能.低功耗.易维护.组管理功能.适 ...

  8. 用kali执行arp攻击-----------使对方断网

    实现原理 其主要原理是局域网内的"攻击机"通过冒充同网络号下的"受害者主机"的物理地址(mac地址),通过欺骗网关,让网关原来应该发给"受害者主机&q ...

  9. 线程池工厂方法newFixedThreadPool()和newCachedThreadPool()

    newFixedThreadPool()方法: 该方法返回一个固定数量的线程池,当一个新的任务提交时,线程池中若有空闲线程,则立即执行. 若没有.则新的任务被暂存在一个任务队列中,待线程空闲时,便处理 ...

  10. 堡垒机paramiko模块

    paramiko简介: 模拟ssh客户端,使用ssh协议,基于sftp协议等做批量管理.例如处理用ssh登陆一千台机器执行同一个命令,或下载上传文件等需求 基于用户名密码登录执行命令: import ...