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() ...
随机推荐
- java基础-day6
第06天 java基础语法 今日内容介绍 u Eclipse断点调试 u 基础语法的练习 第1章 Eclipse断点调试 1.1 Eclipse断点调试概述 Eclipse的断点调试可以 ...
- protobuf和protostuff的区别
在我们的开发过程中,序列化是经常需要处理的问题,比如在做分布式访问数据时,或者是在做redis缓存存储数据时,如果我们涉及的知识面不够广的话,可能会简单的使用JDK的序列化,也即在需要序列化的类上im ...
- 《mysql必知必会》学习_第四章_20180724_欢
P27: select prod_name from products; # select 列 from 表 # 从表products 中检索 名为 prod_name 的列 P28 多个语句一起必须 ...
- Django:全文检索功能可参考博客
https://blog.csdn.net/AC_hell/article/details/52875927 https://www.zmrenwu.com/courses/django-blog-t ...
- 【VB.NET】利用纯真IP数据库查询IP地址及信息
几年前从某个博客抄来的,已经忘记原地址了,如果需要C#版的,可以在博客园搜到吧.我因为自己用,所以转换为了VBNET代码,而且也放置了很久,今天无意间翻出来,就分享给大家吧. 首先,先下载 纯真数据库 ...
- 背水一战 Windows 10 (45) - 控件(图标类): IconElement, SymbolIcon, FontIcon, PathIcon, BitmapIcon
[源码下载] 背水一战 Windows 10 (45) - 控件(图标类): IconElement, SymbolIcon, FontIcon, PathIcon, BitmapIcon 作者:we ...
- 【转】Windows Server 2016 安装 IIS 服务时提示指定备用源路径
原文地址:http://www.codingwhy.com/view/973.html 在Windows Serever 2016中安装IIS的时候,遇到以下提示 是否需要指定备用源路径?一个或多个安 ...
- 回头探索JDBC及PreparedStatement防SQL注入原理
概述 JDBC在我们学习J2EE的时候已经接触到了,但是仅是照搬步骤书写,其中的PreparedStatement防sql注入原理也是一知半解,然后就想回头查资料及敲测试代码探索一下.再有就是我们在项 ...
- centOS7.10 KDE桌面字体设置推荐
安装完centOS7.10的KDE桌面后,第一次使用觉得字体太难看了,特别是终端,看着很难受,调整多次后觉得如下设置舒服很多,分享出来以供参考. 其中等宽字 这样整体看着就会舒服很多 ******** ...
- 解决修改css或js文件,浏览器缓存更新问题。
在搜索引擎中搜索关键字.htaccess 缓存,你可以搜索到很多关于设置网站文件缓存的教程,通过设置可以将css.js等不太经常更新的文件缓存在浏览器端,这样访客每次访问你的网站的时候,浏览器就可以从 ...