接口interface

  • 接口是一个或多个方法签名的集合
  • 只要某个类型拥有该接口的所有方法签名,即算实现该接口,无需显示声明实现了哪个接口,这成为Structural Typing
  • 接口只有方法声明,没有实现,没有数据字段
  • 接口可以匿名嵌入其它接口,或嵌入到结构中
  • 将对象赋值给接口时,会发生拷贝,而接口内部存储的是指向这个复制品的指针,既无法修改复制品的状态,也无法获取指针
  • 只有当接口存储的类型和对象都为nil时,接口才等于nil
  • 接口调用不会做receiver的自动转换
  • 接口同样支持匿名字段方法
  • 接口也可实现类似OOP中的多态
  • 空接口可以作为任何类型数据的容器
package main

import (
"fmt"
) type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
//连接的方法
Connect()
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
var a USB //声明一个变量,它的类型是USB接口
a = PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法取到name 这个字段
a.Connect()
}
package main

import (
"fmt"
) type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
//连接的方法
Connect()
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
// var a USB //声明一个变量,它的类型是USB接口
a := PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法去到name 这个字段
a.Connect()
Disconnect(a) //这里我们将a传进去,会发现a就是USB类型的
}
func Disconnect(usb USB) { // 这里传进去的要求,类型必须是USB的这个接口
fmt.Println("Disconnected.")
}

嵌入接口

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter //这里将connecter的接口嵌入到USB当中
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
// var a USB //声明一个变量,它的类型是USB接口
a := PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法去到name 这个字段
a.Connect()
Disconnect(a) //这里我们将a传进去,会发现a就是USB类型的,所以可以验证PhoneConnecter实现USB的接口
}
func Disconnect(usb USB) { // 这里传进去的要求,类型必须是USB的这个接口
fmt.Println("Disconnected.")
}

通过类型断言的ok pattern可以判断接口中的数据类型

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
// var a USB //声明一个变量,它的类型是USB接口
a := PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法去到name 这个字段
a.Connect()
Disconnect(a) //这里我们将a传进去,会发现a就是USB类型的,所以可以验证PhoneConnecter实现USB的接口
}
func Disconnect(usb USB) { // 这里传进去的要求,类型必须是USB的这个接口
if pc, ok := usb.(PhoneConnecter); ok {
fmt.Println("Disconnected.", pc.name)
return
}
fmt.Println("Unknow device.") }
//go语言当中任何类型都实现了空接口

go语言当中任何类型都实现了空接口

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
// var a USB //声明一个变量,它的类型是USB接口
a := PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法去到name 这个字段
a.Connect()
Disconnect(a) //这里我们将a传进去,会发现a就是USB类型的,所以可以验证PhoneConnecter实现USB的接口
}
func Disconnect(usb interface{}) { // 这里把USB替换成空接口
if pc, ok := usb.(PhoneConnecter); ok {
fmt.Println("Disconnected.", pc.name)
return
}
fmt.Println("Unknow device.") }

使用type switch则可针对空接口进行比较全面的类型判断

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
// var a USB //声明一个变量,它的类型是USB接口
a := PhoneConnecter{"PhoneConnecter"} //这里的a是usb类型的,它无法去到name 这个字段
a.Connect()
Disconnect(a) //这里我们将a传进去,会发现a就是USB类型的,所以可以验证PhoneConnecter实现USB的接口
}
func Disconnect(usb interface{}) { // 这里把USB替换成空接口
switch v := usb.(type) {
case PhoneConnecter:
fmt.Println("Disconnected.", v.name)
default:
fmt.Println("Unknow device.")
} }

接口的相互转换

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
pc := PhoneConnecter{"PhoneConnecter"}
var a Connecter
a = Connecter(pc) //这里的a是强制类型转换,将pc转换成了Connecter接口
a.Connect() //这里可以看到接口的转换,只能由超级转换成子集
}
//可以将拥有超集的接口转换为子集的接口
func Disconnect(usb interface{}) { // 这里把USB替换成空接口
switch v := usb.(type) {
case PhoneConnecter:
fmt.Println("Disconnected.", v.name)
default:
fmt.Println("Unknow device.")
} }

将对象赋值给接口时,会发生拷贝,而接口内部存储的是指向这个

复制品的指针,既无法修改复制品的状态,也无法获取指针

package main

import (
"fmt"
) type Connecter interface {
Connect()
}
type USB interface { //定义一个接口,这个接口下面包含两个方法
//Name方法返回一个字符串
Name() string
Connecter
} type PhoneConnecter struct { //用这个struct对这个USB的接口进行实现
name string
} func (pc PhoneConnecter) Name() string { //实现接口就是用这个结构,实现interface的方法
return pc.name
}
func (pc PhoneConnecter) Connect() {
fmt.Println("Connect:", pc.name)
}
func main() {
pc := PhoneConnecter{"PhoneConnecter"}
var a Connecter
a = Connecter(pc)
a.Connect()
pc.name = "pc" //这里将对象赋值给接口时,会发生拷贝,接口内部存储的是指向这个复制品的指针,
fmt.Println(pc.name) //既无法修改复制品的状态,也无法获取指针
a.Connect()
}
func Disconnect(usb interface{}) { // 这里把USB替换成空接口
switch v := usb.(type) {
case PhoneConnecter:
fmt.Println("Disconnected.", v.name)
default:
fmt.Println("Unknow device.")
} }

Golang 接口interface的更多相关文章

  1. Golang接口(interface)三个特性(译文)

    The Laws of Reflection 原文地址 第一次翻译文章,请各路人士多多指教! 类型和接口 因为映射建设在类型的基础之上,首先我们对类型进行全新的介绍. go是一个静态性语言,每个变量都 ...

  2. 【Golang】Go 通过结构(struct) 实现接口(interface)

    一.通过结构(struct) 实现 接口(interface) 1.在了解iris框架的时候,经常看到有这样去写的使用一个空结构体作为接收器,来调用方法,有点好奇这样做有什么意义. 解释:在 Go 语 ...

  3. Golang的Interface是个什么鬼

    问题概述 Golang的interface,和别的语言是不同的.它不需要显式的implements,只要某个struct实现了interface里的所有函数,编译器会自动认为它实现了这个interfa ...

  4. Go语言学习笔记(四)结构体struct & 接口Interface & 反射

    加 Golang学习 QQ群共同学习进步成家立业工作 ^-^ 群号:96933959 结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struc ...

  5. Go 接口(interface)

        文章转载地址:https://www.flysnow.org/2017/04/03/go-in-action-go-interface.html 1.什么是 interface? 简单的说,i ...

  6. golang的interface剖析

      背景: golang的interface是一种satisfied式的.A类只要实现了IA interface定义的方法,A就satisfied了接口IA.更抽象一层,如果某些设计上需要一些更抽象的 ...

  7. Go语言学习笔记(四)结构体struct & 接口Interface & 反射reflect

    加 Golang学习 QQ群共同学习进步成家立业工作 ^-^ 群号:96933959 结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struc ...

  8. 【Golang 接口自动化06】微信支付md5签名计算及其优化

    前言 可能看过我博客的朋友知道我主要是做的支付这一块的测试工作.而我们都知道现在比较流行的支付方式就是微信支付和支付宝支付,当然最近在使用低手续费大力推广的京东金融(已改名为京东数科)以后也可能站到第 ...

  9. Golang 接口与反射知识要点

    目录 Golang 接口与反射知识要点 1. 接口类型变量 2. 类型断言 3. 鸭子类型 4. 反射机制 5. reflect 包 TypeOf().ValueOf() Type().Kind() ...

随机推荐

  1. Hadoop/Spark相关面试问题总结

    面试回来之后把其中比较重要的问题记了下来写了个总结: (答案在后面) 1.简答说一下hadoop的map-reduce编程模型 2.hadoop的TextInputFormat作用是什么,如何自定义实 ...

  2. 《mysql必知必会》学习_第10章_20180731_欢

    第10章,计算字段. P64 select concat (vend_name,'(',vend_country,')') from vendors order by vend_name; # 拼接, ...

  3. poj 3013 最短路变形

    http://poj.org/problem?id=3013 给出n个点,m个边.给出每个点的权值,每个边的权值.在m条边中选n-1条边使这n个点成为一棵树,root=1,求这棵树的最小费用,费用=树 ...

  4. 一个前端小白,关于vue\react等框架下table的应用总结

    出来实习一个月多,对于前端,运用相关的最多的就是table,想总结一下先关的内容 一.table提供的功能 1.显示表 2.可编辑:分为可编辑行和可编辑块,但是原理都一样就是设置一个flag,true ...

  5. Maven的依赖管理

    我们知道dependencies是可以被继承的,这个时候我们就想到让我们的发生了共用的依赖元素转移到parent中,这样我们又进一步的优化了配置.可是问题也随之而来,如果有一天我创建了一个新的模块,但 ...

  6. unigui的ServerModule常用属性设置

    unigui的ServerModule常用属性设置 1)压缩设置 compression是压缩数据用的.默认启用压缩,且压缩级别是最大的. 2)UNIGUI运行时库设置 UNIGUI需要4个运行时库, ...

  7. cxGrid单元格获得输入焦点

    cxGrid单元格获得输入焦点   cxGrid单元格获得输入焦点 cxGrid1.SetFocus;cxGrid1DBTableView1.Controller.EditingController. ...

  8. android 获取 imei号码

    Imei = ((TelephonyManager) getSystemService(TELEPHONY_SERVICE)) .getDeviceId(); 1.加入权限 在manifest.xml ...

  9. asp.net core 健康检查

    asp.net core 健康检查 ASP.NET Core 2.2 开始,提供了健康检查中间件和库,用来报告应用基础结构组件的运行状况.官方文档在此 运行状况检查由应用程序作为 HTTP 终结点公开 ...

  10. EF学习笔记-2 EF之支持复杂类型的实现

    使用过.NET的小伙伴们知道,在我们的实体模型中,除了一些简单模型外,还有一些复杂类型,如几个简单的类型组合而成的类型:而EF除了在实现基本的增删改查之外,也支持复杂类型的实现. 那么如何手动构造复杂 ...