接口作用

   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. 阿里Java中级开发测试相关试题及答案

    ------------恢复内容开始------------ 1.JSP文件上传的HTML标签代码定义是? A. <input type="text" name=" ...

  2. laya资源版本管理

    layaair-cmd https://ldc.layabox.com/doc/?nav=zh-as-3-4-2 Laya的版本管理 https://blog.csdn.net/qq_43287088 ...

  3. Linux—账号管理及命令使用详解

    关注微信公众号:CodingTechWork,一起学习进步. 引言   在Linux系统中,我们常常会看到目录或文件的所属关系: [root@linux01 ~]# ll -d test.sh -rw ...

  4. vue-devtools-4.1.4_0.crx及Vue.js not detected的问题

    谷歌-更多工具-扩展程序 Vue.js not detected的问题

  5. python urllib.request

    一.简介 urllib.request 模块提供了访问 URL 的相关功能 二.常用函数 urllib.request.urlopen("http://httpbin.org", ...

  6. [03] C# Alloc Free编程

    C# Alloc Free编程 首先Alloc Free这个词是我自创的, 来源于Lock Free. Lock Free是说通过原子操作来避免锁的使用, 从而来提高并行程序的性能; 与Lock Fr ...

  7. Linux:文件和用目录权限基础

    首先,用ls -i 查看文件的信息: 红框框起来的部分:-rw-r--r-- 一共是10位,第一位表示文件类型,-表示文件,d表示文件夹/目录:后面9位则表示权限 其中前三位是该文件/目录所属者的权限 ...

  8. 如何编写一个简单的Linux驱动(二)——设备操作集file_operations

    前期知识 如何编写一个简单的Linux驱动(一)--驱动的基本框架 前言 在上一篇文章中,我们学习了驱动的基本框架.这一章,我们会在上一章代码的基础上,继续对驱动的框架进行完善.要下载上一篇文章的全部 ...

  9. 大神Java8写了一段逻辑,我直呼看不懂

    业务背景 首先,业务需求是这样的,从第三方电商平台拉取所有订单,然后保存到公司自己的数据库,需要判断是否有物流信息,如果有物流信息,还需要再进行上传. 而第三方接口返回的数据是 JSON 格式的,其中 ...

  10. 教你一招Linux下文本比对方法

    我们在写代码的过程中,免不了会对代码进行一些修修改改.但经常会出现改着改着,就不知道改完后与源文件的差异是怎样的.这里,我们就需要一个文本比对工具来进行文本比对. 有经验的程序员都知道,Windows ...