Golang 接口interface
接口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的更多相关文章
- Golang接口(interface)三个特性(译文)
The Laws of Reflection 原文地址 第一次翻译文章,请各路人士多多指教! 类型和接口 因为映射建设在类型的基础之上,首先我们对类型进行全新的介绍. go是一个静态性语言,每个变量都 ...
- 【Golang】Go 通过结构(struct) 实现接口(interface)
一.通过结构(struct) 实现 接口(interface) 1.在了解iris框架的时候,经常看到有这样去写的使用一个空结构体作为接收器,来调用方法,有点好奇这样做有什么意义. 解释:在 Go 语 ...
- Golang的Interface是个什么鬼
问题概述 Golang的interface,和别的语言是不同的.它不需要显式的implements,只要某个struct实现了interface里的所有函数,编译器会自动认为它实现了这个interfa ...
- Go语言学习笔记(四)结构体struct & 接口Interface & 反射
加 Golang学习 QQ群共同学习进步成家立业工作 ^-^ 群号:96933959 结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struc ...
- Go 接口(interface)
文章转载地址:https://www.flysnow.org/2017/04/03/go-in-action-go-interface.html 1.什么是 interface? 简单的说,i ...
- golang的interface剖析
背景: golang的interface是一种satisfied式的.A类只要实现了IA interface定义的方法,A就satisfied了接口IA.更抽象一层,如果某些设计上需要一些更抽象的 ...
- Go语言学习笔记(四)结构体struct & 接口Interface & 反射reflect
加 Golang学习 QQ群共同学习进步成家立业工作 ^-^ 群号:96933959 结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struc ...
- 【Golang 接口自动化06】微信支付md5签名计算及其优化
前言 可能看过我博客的朋友知道我主要是做的支付这一块的测试工作.而我们都知道现在比较流行的支付方式就是微信支付和支付宝支付,当然最近在使用低手续费大力推广的京东金融(已改名为京东数科)以后也可能站到第 ...
- Golang 接口与反射知识要点
目录 Golang 接口与反射知识要点 1. 接口类型变量 2. 类型断言 3. 鸭子类型 4. 反射机制 5. reflect 包 TypeOf().ValueOf() Type().Kind() ...
随机推荐
- noip第12课作业
1. 输出低于班级平均分的学生信息 [问题描述] 输入N个学生的姓名和语文.数学的得分,求平均分低于班级平均分的学生,将其信息全部输出.分数相同的按输入先后输出. 输入格式:第1行,有一个整数N ...
- Tarjan求缩点化强连通图
Describe: 求一个有向图加多少条边可以变成一个强连通图 Solution: Tarjan缩点染色后,判断出度和入度,所有点的出度 = 0 的和 和 入度 = 0 的和的最大值即为所求. 缩点染 ...
- Hive Bug修复:ORC表中array数据类型长度超过1024报异常
目前HVIE里查询如下语句报错: select * from dw.ticket_user_mtime limit 10; 错误如下: 17/07/06 16:45:38 [main]: DEBUG ...
- 【UOJ244】 【UER #7】短路(贪心)
传送门 uoj Solution 简单题? 考虑一条路径长什么样子,一定是经过某一个字母环的左上角,那么答案就很简单了. 我们记一个前缀最小值,这样子让他一路走下去一定是最优! 然后扫一遍就好了. 代 ...
- [BeiJing wc2012]连连看(建模,最小费用最大流)
前言 突然发现自己在图论①被dalao吊着打... Solution 看到数据范围1000,感觉可以直接枚举连边,然后新建两个点就好了. 注意要拆点,不然可能会死循环(过来人) 代码实现 #inclu ...
- Git错误日志--! [rejected]
当出现下面错误时 ,是因为和仓库两端版本不一致时,常见于刚创建仓库时,中只有readme文件的情况.执行下面的运行完毕之后.再push到仓库即可 ! [rejected] master -> m ...
- Linux的1000个命令
目录 Linux常用命令 uptime wget uname free who last history pwd cd ls cat head tail tr wc cut diff touch mk ...
- 二进制入门-打造Linux shellcode基础篇
0x01 前言 本文的目的不是为了介绍如何进行恶意的破坏性活动,而是为了教会你如何去防御此类破坏性活动,以帮助你扩大知识范围,完善自己的技能,如有读者运用本文所学技术从事破坏性活动,本人概不负责. ...
- GC日志时间分析
在GC日志里,一条完整的GC日志记录最后,会带有本次GC所花费的时间,如下面这一条新生代GC: [GC [DefNew: 3298K->149K(5504K), secs] [Times: us ...
- mybatis 插件原理
[传送门]:mybatis 插件原理