参考链接 https://blog.csdn.net/tzs919/article/details/53571632

type是golang中非常重要的关键字,常见的就是定义结构体,但是其功能远不止是像c中那样只定义结构体,在golang中type关键字的功能可以说是非常丰富,通过参考相关的文章和源码,总结如下:

1 定义结构体

type person struct {
name string //注意后面不能有逗号
age int
}

2 类型定义,相当于定义一个别名

type name string   //name类型与string等价
例子:
type name string func main() {
var myname name = "taozs" //其实就是字符串类型
l := []byte(myname) //字符串转字节数组
fmt.Println(len(l)) //字节长度
}

ps:定义的新类型(别名),可以用来定义方法,比如,我设置string的别名为name,可以用name定义方法,如下:

type name string

func (n name) len() int {
return len(n)
} func main() {
var myname name = "taozs" //其实就是字符串类型
l := []byte(myname) //字符串转字节数组
fmt.Println(len(l)) //字节长度
fmt.Println(myname.len()) //调用对象的方法
}

3 type定义结构体时,可以内嵌匿名成员

//结构体内嵌匿名成员定义

type person struct {
string //直接写类型,匿名
age int
} func main() {
//结构体匿名成员初始化
p := person{string: "taozs", age: 18}//可以省略部分字段,如:person{string: "taozs"}。也可以这样省略字段名:person{“taozs”, 18},但必须写全了,不可以省略部分字段
//结构体匿名成员访问
fmt.Println(p.string) //注意不能用强制类型转换(类型断言):p.(string)
}

当结构体只有唯一一个匿名成员的case时

//结构体内嵌匿名成员定义
type person struct {
string
} func main() {
//结构体匿名成员初始化
p := person{string: "taozs"} //也可这样:person{"taozs"}
//结构体匿名成员访问
fmt.Println(p.string) //注意不能用强制类型转换(类型断言):p.(string)
}

4 定义接口类型

这也是最常用的一种,例子如下,不多赘述

//接口定义
type Personer interface {
Run()
Name() string
}
//实现接口,注意实现接口的不一定只是结构体,也可以是函数对象,参见下面第5条
type person struct {
name string
age int
} func (person) Run() {
fmt.Println("running...")
} //接收参数person不可以是指针类型,否则不认为是实现了接口 func (p person) Name() string {
return p.name
} func main() {
//接口类型的变量定义
var p Personer
fmt.Println(p) //值<nil>
//实例化结构体,并赋值给interface
p = person{"taozs", 18} //或者:&person{"taozs", 18}
p.Run()
fmt.Println(p.Name())
var p2 person = p.(person) //类型断言,接口类型断言到具体类型
fmt.Println(p2.age)
}

ps:当一个结构体实现了某个接口的所有方法时,才能算是实现了这个接口

//另外,类型断言返回值也可以有第二个bool值,表示断言是否成功,如下:
if p2, ok := p.(person); ok {//断言成功ok值为true
fmt.Println(ok)
fmt.Println(p2.age)
}

5 定义函数类型

//以下是定义一个函数类型handler
type handler func(name string) int //针对这个函数类型可以再定义方法,如:
func (h handler) add(name string) int {
return h(name) + 10
}

 

下面是一个比较全面的例子:

package main

import (
"fmt"
) //定义接口
type adder interface {
add(string) int
} //定义函数类型
type handler func(name string) int //实现函数类型方法
func (h handler) add(name string) int {
return h(name) + 10
} //函数参数类型接受实现了adder接口的对象(函数或结构体)
func process(a adder) {
fmt.Println("process:", a.add("taozs"))
} //另一个函数定义
func doubler(name string) int {
return len(name) * 2
} //非函数类型
type myint int //实现了adder接口
func (i myint) add(name string) int {
return len(name) + int(i)
} func main() {
//注意要成为函数对象必须显式定义handler类型
var my handler = func(name string) int {
return len(name)
} //以下是函数或函数方法的调用
fmt.Println(my("taozs")) //调用函数
fmt.Println(my.add("taozs")) //调用函数对象的方法
fmt.Println(handler(doubler).add("taozs")) //doubler函数显式转换成handler函数对象然后调用对象的add方法
//以下是针对接口adder的调用
process(my) //process函数需要adder接口类型参数
process(handler(doubler)) //因为process接受的参数类型是handler,所以这儿要强制转换
process(myint(8)) //实现adder接口不仅可以是函数也可以是结构体 }

 难点:fmt.Println(handler(doubler).add("taozs")) //doubler函数显式转换成handler函数对象然后调用对象的add方法

  这波操作比较懵逼,首先说下这波操作一下子就能看懂的,看注释,很明显,强制类型转换,doubler函数强制转换成了handler这种类型,这个操作最后的输出是20,怎么得来的呢?看add方法的实现:

func (h handler) add(name string) int {
return h(name) + 10
}

   可以看到,像是声明某个结构体的方法的方式,这里规定了add属于handler这个函数类型,可以看到add函数体中有一个h(name),是不是看的有点莫名其妙?其实就像是接口一样,哪个句柄调用了add方法,就执行哪个句柄。可以看到

  fmt.Println(handler(doubler).add("taozs"))

  是doubler调用了add,那么这个h(name)就等价于doubler(name),因为这里name="taozs",那么先执行add函数,执行到return这句的时候,执行h(name)即doubler(name),执行结果是len(name)*2 = 10。于是,结果就是20

golang type的更多相关文章

  1. golang type 和断言 interface{}转换

    摘要 类型转换在程序设计中都是不可避免的问题.当然有一些语言将这个过程给模糊了,大多数时候开发者并不需要去关 注这方面的问题.但是golang中的类型匹配是很严格的,不同的类型之间通常需要手动转换,编 ...

  2. golang type conversion

    map[string]interface{} is not the same as map[string]string. Type interface{} is not the same as typ ...

  3. golang编程之我见

    golang编程之我见 学习了两个月的golang,语法算是基本掌握了,从一个C++程序员的角度,提出自己的几个看法吧. 1,没有一个好的包管理工具. 我在公司用的是glide的包管理,这个工具的好处 ...

  4. nginx-unit docker 运行以及php &&golang 简单使用

    备注:    nginx  unit  nginx 开源的新的nginx 开发平台,但是说白了,个人感觉一般,而且官方文档也不是很好, api 接口目前暂时文档比较....,以前写过虚拟机&& ...

  5. Golang 基础语法介绍及对比(二)

    传值与传参 Golong func main() { a := fmt.Println("a = ", a) // 应该输出 "a= 3" a1 := add1 ...

  6. C、C++、Java、go的语法区别

    详细C++.Java比较:http://www.cnblogs.com/stephen-liu74/archive/2011/07/27/2118660.html 一.C.C++的区别 在很大程度上, ...

  7. VSCode配置Go插件和第三方拓展包

    前言 VSCode现在已经发展的相当完善,很多语言都比较推荐使用其来编写,Go语言也一样,前提你电脑已经有了Go环境和最新版本的VSCode 插件安装 直接在拓展插件中搜索Go,就可以安装Go插件 安 ...

  8. 广度优先搜索(BFS)解题总结

    定义 广度优先搜索算法(Breadth-First-Search),是一种图形搜索算法. 简单的说,BFS是从根节点开始,沿着树(图)的宽度遍历树(图)的节点. 如果所有节点均被访问,则算法中止. B ...

  9. 最小生成树MST算法(Prim、Kruskal)

    最小生成树MST(Minimum Spanning Tree) (1)概念 一个有 n 个结点的连通图的生成树是原图的极小连通子图,且包含原图中的所有 n 个结点,并且有保持图连通的最少的边,所谓一个 ...

随机推荐

  1. python基础学习1-随机验证码

    import random i=random.randrange(65,90) #根据设置的范围生成随机数字 print(i) c=chr(i)#根据随机产生的数字 然后用chr生成对应ASCII 数 ...

  2. CF1017G The Tree 树链剖分

    CF1017G The Tree LG传送门 树链剖分好题. 乍一看还以为是道沙比题,然后发现修改操作有点不一样. 但是如果你对基本操作还不太熟练,可以看看我的树链剖分总结 有三个操作: 从一个点往下 ...

  3. 使用cJSON库解析JSON

    cJSON库的下载 cJSON是一个基于C的JSON解析库,这个库非常简单,只有cJSON.c和cJSON.h两个文件,支持JSON的解析和封装,需要调用时,只需要#include "cJS ...

  4. directive指令二 require:'^ngModel'

    本章主要是讲指令与ngModel的交互. 在angular有一个内置指令叫ngModel,它是angular用来处理表单的最重要的指令.在源码中,页面上的model值的格式化.解析.验证都是由ngMo ...

  5. Git命令简单总结

    集中式vs分布式 svn集中式:版本库是集中存放在中央服务器的,需要联网才能工作 git 分布式:每个人的电脑上都是一个完整的版本库 和集中式版本控制系统相比,分布式版本控制系统的安全性要高很多,因为 ...

  6. JMeter的__threadGroupName使用注意事项

    JMeter从4.1版本开始引入了一个新函数"${__threadGroupName}",这个函数的作用是返回当前线程组的名字.${__threadGroupName}的用途也较为 ...

  7. 统计学习方法c++实现之八 EM算法与高斯混合模型

    EM算法与高斯混合模型 前言 EM算法是一种用于含有隐变量的概率模型参数的极大似然估计的迭代算法.如果给定的概率模型的变量都是可观测变量,那么给定观测数据后,就可以根据极大似然估计来求出模型的参数,比 ...

  8. lua字符串类型

    Lua中字符串结构体的定义是: typedef union TString { L_Umaxalign dummy; /* ensures maximum alignment for strings ...

  9. 人工智能AI芯片与Maker创意接轨 (中)

    在人工智能AI芯片与Maker创意接轨(上)这篇文章中,介绍人工智能与深度学习,以及深度学习技术的应用,了解内部真实的作业原理,让我们能够跟上这波AI新浪潮.系列文来到了中篇,将详细介绍目前市面上的各 ...

  10. trampoline蹦床函数解决递归调用栈问题

    递归函数的调用栈太多,造成溢出,那么只要减少调用栈,就不会溢出.怎么做可以减少调用栈呢?就是采用"循环"换掉"递归". 下面是一个正常的递归函数. functi ...