接口作用

   Go语言中的接口是一种类型,类似于Python中的抽象基类。

   Go语言中使用接口来体现多态,是duck-type的一种体现。

   如,只要一个东西会叫,会走,那么我们就可以将它定义为一个动物的接口。

接口定义

   Go中提倡面向接口编程,以下是接口的定义。

type 接口类型名 interface{
方法名1( 参数列表1 ) 返回值列表1
方法名2( 参数列表2 ) 返回值列表2

}

   关于接口的定义有以下几点注意事项:

   接口名在单词后面一般都需添加er的后缀,代表这是一个接口

   当接口名与方法名都是大写时,代表该接口与方法均可被外部包进行访问

   参数列表以及返回值列表参数变量名可以省略

   以下我们将定义一个动物的接口,会叫,会移动我们将将它看作为动物。

   并且我们为该接口定义了一个方法撒泼,只要是动物就可以进行撒泼。

package main

import (
"fmt"
) type animal interface {
move()
roar()
} func sapo(a animal){ // 接收一个动物类型
a.move()
a.roar()
}

接口使用

   如何使用上面的接口呢?其实我们还需要做结构体。

   如下示例,做了一个狗的结构体和一个狼的结构体并且进行实例化,由于狗和狼都实现了move以及roar方法,所以这两个实例化都可以看作是animal类型,即可以调用sapo方法。

package main

import (
"fmt"
) type animal interface {
move()
roar()
} func sapo(a animal){
a.move()
a.roar()
} type dog struct {
name string } func (d dog) move() {
fmt.Printf("%s在移动\n", d.name)
} func (d dog) roar() {
fmt.Printf("%s在吼叫\n", d.name)
} type wolf struct {
name string } func (w wolf) move() {
fmt.Printf("%s在移动\n", w.name)
} func (w wolf) roar() {
fmt.Printf("%s在吼叫\n", w.name)
} func main() {
d1 := dog{
name: "大黄",
}
w1 := wolf{
name: "灰太狼",
}
sapo(d1) // 大黄调用动物的撒泼方法,由于会动会叫就是动物,所以大黄有两种类型,一种是动物,一种是狗
sapo(w1)
} // 大黄在移动
// 大黄在吼叫
// 灰太狼在移动
// 灰太狼在吼叫

接口变量

   接口是一种类型,所以一个变量可以定义为该类型。

   如下所示,声明了一个动物类型的接口变量,当一个结构体实例对象拥有了moveroar方法后,才可成功进行赋值。

package main

import (
"fmt"
) type animal interface {
move()
roar()
} type dog struct {
name string
} func (d dog) move() {
fmt.Printf("%s在移动\n", d.name)
} func (d dog) roar() {
fmt.Printf("%s在吼叫\n", d.name)
} func main() {
var a1 animal // 动物类型
d1 := dog{
name: "大黄",
}
a1 = d1 // 由于大黄有move与roar方法,所以它也算是动物类型。因此可以赋值成功
a1.move()
}

结构体方法类型

   结构体不同的方法类型,会对接口产生不同的影响。

值接收者方法

   当结构体的方法是值接收者方法时,该结构体实例化可以赋值给对应的接口变量,并且是任意形式。

package main

import (
"fmt"
) type people interface {
combHair() // 能梳头就是人
} type person struct {
name string
} // 值接收者方法
func (p person) combHair() {
fmt.Printf("%s在梳头发", p.name)
} func main() {
var pe people // 人类
var p1 = person{
name: "云崖",
} // 全部都可以做为人类
pe = p1
pe = &p1
pe = *(&p1)
fmt.Println(pe)
}

指针接收者方法

   当结构体的方法是指针接收者方法时,该结构体实例化赋值给接口变量时只能是地址传递。

package main

import (
"fmt"
) type people interface {
combHair() // 能梳头就是人
} type person struct {
name string
} // 指针接收者方法
func (p *person) combHair() {
fmt.Printf("%s在梳头发", p.name)
} func main() {
var pe people // 人类
var p1 = person{
name: "云崖",
} pe = *p1 // 错误
pe = &p1 // 只能传递地址
pe = *(&p1) // 错误
fmt.Println(pe)
}

类型与接口

一个类型多个接口

   如会梳头发是人类,会移动是动物类。

   那么我们就可以对person这个结构体做两个接口,一个是人类的接口,一个是动物类的接口。

   person实例化对象p1同时满足人类和动物这两个接口。

package main

import (
"fmt"
) type people interface {
combHair() // 能梳头就是人
} type animal interface {
move() // 能移动就是动物
} type person struct {
name string
} func (p person) move() {
fmt.Println("人移动了")
} func (p person) combHair(){
fmt.Println("人在梳头发")
} func main() {
var pe people // 人类接口变量
var an animal // 动物接口变量
var p1 = person{ // 实例化出一个人
name: "云崖",
} pe = p1
an = p1
fmt.Println(pe)
fmt.Println(an)
}

一个接口多个类型

   如会动的都是动物,那么下面例子中狗和牛等结构体的实例化就都是动物。

   这就是一个动物接口可以有多个类型的体现。

package main

import (
"fmt"
) type people interface {
combHair() // 能梳头就是人
} type animal interface {
move() // 能移动就是动物
} type dog struct {
name string
} type cattle struct {
name string
} func (d dog) move() {
fmt.Println("狗在动")
} func (c cattle) move() {
fmt.Println("牛在动")
} func main() {
var an animal // 动物接口变量 d1 := dog{
name: "大黄",
}
c1 := cattle{
name: "牛魔王",
}
an = d1
fmt.Println(an)
an = c1
fmt.Println(an)
}

接口嵌套

   比如两栖动物可以在水中生活,也可以在陆地生活。

   我们定义一个陆地生活动物的接口,再定义一个水中生活动物的接口。

   如何表示两栖动物的接口呢?只需要应用嵌套让两栖动物的接口同时拥有陆地生活动物接口的方法和水中生活动物的方法即可。

   如下示例:

package main

import (
"fmt"
) type terrestrialAnimal interface {
landMobile() // 陆地移动,则是陆生动物
} type aquatic interface {
swim() // 能够游泳,则是水生动物
} type amphibian interface {
terrestrialAnimal
aquatic
// 同时实现了陆生动物和水生动物的特性,则是两栖动物
} type frog struct{
name string
// 青蛙结构体
} // 青蛙会游泳
func (f frog) swim(){
fmt.Println("青蛙在游泳")
} // 青蛙能上岸
func (f frog) landMobile(){
fmt.Println("青蛙在陆地上欢快的蹦跶")
} func main() {
var amp amphibian // 两栖动物接口变量
var f1 = frog {
name : "呱呱娃",
}
amp = f1 // 青蛙是两栖动物
fmt.Println(amp) }

空接口

   空接口在实际开发中使用非常广泛,它代表可以是任意类型的变量。

接口定义

   使用interface{}来定义一个空接口。

   如下所示:

package main

import (
"fmt"
) func main(){
var arbitraryType interface{} // 接收任意类型的变量
var str = "HELLO,WORLD"
arbitraryType = str
fmt.Printf("%T \n", arbitraryType) // string
var num = int64(100)
arbitraryType = num
fmt.Printf("%T \n",arbitraryType) // int64 }

实际应用

   空接口一般应用于函数形参,以及map值中。

   如下,该函数允许传递任何数据类型的数据。

package main

import (
"fmt"
) func test(v1 interface{}) {
fmt.Printf("%T \n", v1) // int
} func main() {
test(100)
}

   其实更多的应用场景在map中,如下定义的map值可以是任意类型,这样存取都会变得很方便。

package main

import (
"fmt"
) var m = make(map[string]interface{}, 30) func main() {
m["k1"] = "第一个"
m["k2"] = 2
m["k3"] = []int16{1,2,3,4}
fmt.Print(m)
}

类型断言

   由于interface{}空接口可以存储任意类型的值,那么我们如何判断出值的类型?

   方法如下:

x.(T)

   x:表示类型为interface{}的变量

   T:表示断言x可能是的类型。

   方法断言的作用其实就类似于猜,这种用的不是很多。

   如下示例,使用switch进行类型断言的判断:

package main

import (
"fmt"
) var m = make(map[string]interface{}, 30) func test(v1 map[string]interface{}) {
for _, value := range v1 {
switch value.(type) { // 使用断言获取到类型
case string:
fmt.Println("这是一个字符串类型", value)
case int:
fmt.Println("这是一个int类型", value)
case []int16:
fmt.Println("这是一个int16的切片类型", value)
default:
fmt.Println("未识别的类型", value)
}
}
} func main() {
m["k1"] = "第一个"
m["k2"] = 2
m["k3"] = []int16{1, 2, 3, 4}
test(m)
}

Go 接口类型的更多相关文章

  1. 无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接口类型“Microsoft.Office.Interop.Word._Application”。

    无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接口类型“Microsoft.Office.Interop.Wor ...

  2. (转)无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接口类型“Microsoft.Office.Interop.Word._Application”。此操作失败的原因是对 IID 为“{00020970-

    HRESULT:0x80030002 无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接口类型“Microsoft ...

  3. 无法将类型为 excel.applicationclass 的 com 强制转换为接口类型 的解决方法。

    今天碰到客户的电脑在导出EXCEL的时候提示,无法将类型为 excel.applicationclass 的 com 强制转换为接口类型 excel._application 的问题 最后用下面的方法 ...

  4. C#与excel互操作的错误无法将类型为“Excel.ApplicationClass”的COM 对象强制转换为接口类型“Excel._Application”

    如果您使用的电脑要操作的是office2003而之前使用过office2007使用此方法可解决您的问题 无法将类型为“Microsoft.Office.Interop.Excel.Applicatio ...

  5. 无法将类型为“System.__ComObject”的 COM 对象强制转换为接口类型,原因为没有注册类

    错误描述 e = {"无法将类型为"System.__ComObject"的 COM 对象强制转换为接口类型"OpcRcw.Da.IOPCServer" ...

  6. System.InvalidCastException: 无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接口类型“Microsoft.Office.Interop.Word._Application”。

    报错:System.InvalidCastException: 无法将类型为“Microsoft.Office.Interop.Word.ApplicationClass”的 COM 对象强制转换为接 ...

  7. 无法将类型为excel.applicationclass的com 强制转换为接口类型的解决方法[转]

    c#解决方案EXCEL 导出 今天碰到客户的电脑在导出EXCEL的时候提示,无法将类型为 excel.applicationclass 的 com 强制转换为接口类型 excel._applicati ...

  8. Go基础系列:接口类型断言和type-switch

    接口类型探测:类型断言 接口实例中存储了实现接口的类型实例,类型的实例有两种:值类型实例和指针类型实例.在程序运行过程中,接口实例存储的实例类型可能会动态改变.例如: // ins是接口实例 var ...

  9. Go语言规格说明书 之 接口类型(Interface types)

    go version go1.11 windows/amd64 本文为阅读Go语言中文官网的规则说明书(https://golang.google.cn/ref/spec)而做的笔记,介绍Go语言的  ...

  10. mocha测试接口类型及测试报告收集

    记录参考: 参考文档: 测试报告以及es6: http://www.ruanyifeng.com/blog/2015/12/a-mocha-tutorial-of-examples.html 测试接口 ...

随机推荐

  1. 攻防世界——web新手练习区解题总结<2>(5-8题)

    第五题cookie: 所需工具:burpsuite(需自行下载) 老规矩看完题,先获取在线场景,得到如下网页 那么什么是cookie呢?大体上就是网站为了识别用户身份而储存在用户本地终端上的数据,类型 ...

  2. js扩展运算符(spread)三个点(...)

    常见用法: 1.该运算符主要用于函数调用. function push(array, ...items) { array.push(...items); } function add(x, y) { ...

  3. POJ-3255-Roadblocks(次短路的另一种求法)

    Bessie has moved to a small farm and sometimes enjoys returning to visit one of her best friends. Sh ...

  4. CSS中的包含块

    1.初始包含块,浏览器viewport大小 2.非根元素,position:relative/static,包含块为最近的块级框,表格单元或行内祖先框的内容区 3.非根元素,position:abso ...

  5. Vue事件绑定原理

    Vue事件绑定原理 Vue中通过v-on或其语法糖@指令来给元素绑定事件并且提供了事件修饰符,基本流程是进行模板编译生成AST,生成render函数后并执行得到VNode,VNode生成真实DOM节点 ...

  6. node中间件

    npm i body-parser post 请求主题中间件 const bodyParser = require('body-parser')   const bodyParser = requir ...

  7. 浅析vue的两项原理

    一.vue双向绑定原理 Vue.js-作者为中国人尤雨溪 vue实现数据双向绑定主要是:采用数据劫持结合发布者-订阅者模式的方式,通过Object.defineProperty()来劫持各个属性的se ...

  8. LAMP 和 LNMP

    #0x01 组成: LAMP==Linux+Apache+Mysql+PHP LNMP==Linux+Nginx+Mysql+PHP LANMP==linux + nginx + apache + m ...

  9. leetcode刷题-71简化路径

    题目 以 Unix 风格给出一个文件的绝对路径,你需要简化它.或者换句话说,将其转换为规范路径. 在 Unix 风格的文件系统中,一个点(.)表示当前目录本身:此外,两个点 (..) 表示将目录切换到 ...

  10. pytest文档1-pytest+Allure+jenkins+邮箱发送

    前言: 1.pytest+allure是目前很多公司使用较多的一种报告样式,因为它更详细,各种指标更直观(简单的说就是看着更高大上,更能装X). 环境准备: 1.Windows10 2.Allure ...