接口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. Pyenv 安装部署

    Pyenv Pyenv是个多版本Python管理器,可以同时管理多个Python版本共存, 区别于virtualenv. 安装 git clone git://github.com/yyuu/pyen ...

  2. poj 2449 k短路+A*算法

    http://poj.org/problem?id=2449 K短路的定义: 1.如果起点终点相同,那么0并不是最短路,而是要出去一圈回来之后才是最短路,那么第K短路也是一样. 2.每个顶点和每条边都 ...

  3. PHP搜索 搜索 搜索

    //搜索界面 public function search(){ $param=input('param.'); $where=[]; //搜索框 if(!empty($param['content' ...

  4. AngularJS 指令中的require

    require参数可以被设置为字符串或数组,字符串代表另外一个指令的名字.require会将控制器注入到其值所指定的指令中,并作为当前指令的链接函数的第四个参数.字符串或数组元素的值是会在当前指令的作 ...

  5. scikit-FEM-mesh

    # -*- coding: utf-8 -*- """ “Mesh”模块包含了有限元网格的不同类型. See the following implementations: ...

  6. Delphi XE3写DLL,用Delphi7调用,报错!

    http://bbs.csdn.net/topics/390870532 用delphi xe3写的DLL,delphi7调用,参数都是PAnsiChar,DLL里的函数接收delphi7传的入参,没 ...

  7. 我最常用的7个Web在线工具

    为什么要用Web在线工具呢?有两个原因,第一,它不受限于物理平台,我既可以在自己的电脑上使用,也可以在公司或亲戚朋友的电脑上使用(不管对方的操作系统是什么,只要能上网):第二,可以解放硬盘,减少PC端 ...

  8. Redis .Net

    一.Redis简介 Redis是一个开源的使用ANSI C语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value数据库,和Memcached类似,它支持存储的value类型相对更多,包括 ...

  9. [luogu 5301][bzoj 5503] [GXOI/GZOI2019] 宝牌一大堆

    题面 好像ZJOI也考了一道麻将, 这是要发扬中华民族的赌博传统吗??? 暴搜都不会打, 看到题目就自闭了, 考完出来之后看题解, \(dp\), 可惜自己想不出来... 对于国士无双(脑子中闪过了韩 ...

  10. soap注入某sql2008服务器结合msf进行提权

    原文作者:陈小兵 在实际成功渗透过程中,漏洞的利用都是多个技术的融合,最新技术的实践,本次渗透利用sqlmap来确认注入点,通过sqlmap来获取webshell,结合msf来进行ms16-075的提 ...