前言

  在go语言中,type用于类型定义(type definition)与类型别名(type alias)。这两者的差别从名字上已经可以初见端倪。

  类型定义即定义新类型,是一个全新的类型,但可能与被定义类型存在一些关系,如类型转换,方法继承等。类型定义在各语言中有不同的体现,如Java是用class, interface等关键字作类型定义,在python中也是使用class关键字。只不过go为了简洁的原则,可以省些关键字,因而与类型别名重用了type关键字。

  类型别名则是对被定义类型的别称,与其是相同的类型,只不过取了另外一个名字而已。类型别名是本文讨论的重点。这里有几个问题,一是为何go要引入类型别名,其有什么好处?为什么Java或者其他语言没有类型别名?

  在go语言中,type有5种作用,罗列如下。但在本文中,只讲述类型定义与类型别名。

  1. 定义接口
  2. 定义结构体
  3. 类型定义
  4. 类型别名
  5. 类型查询

类型定义

package main

import "fmt"

func main() {
h := House{height: 1.0, width: 2.0}
fmt.Println(h.Height())
wh := WhiteHouse{height: 1.0, width: 2.0}
fmt.Println(wh.Area())
w := House(wh) //类型转换
fmt.Println(w.Width())
f1 := Factory{height: 1.0, width: 2.0}
f2 := Factory2{name: "test"}
fw := House(f1)
fmt.Println(fw)
fw2 := House(f2) //运行时这里会报类型转换错误
fmt.Println(fw2)
} type House struct {
height float32
width float32
} func (h *House) Height() float32 {
return h.height
} func (h *House) Width() float32 {
return h.width
} type WhiteHouse House //类型定义 func (h *WhiteHouse) Area() float32 {
return h.height * h.width
} type Factory struct {
height float32
width float32
} type Factory2 struct {
name string
}

  WhiteHouse与House是两个不同的类型,方法Area只属于WhiteHouse。 WhiteHouse可以转换为House类型。但对于类型Factory与Factory2,House与WhiteHouse都只能与Factory进行类型转换(尽管这种类型转换在实践中基本不会出现),而与Factory2进行类型转换则会报错。个中原因在于struct类型的内存布局是按顺序存储的,因为只要类型拥有的属性一致,则进行类型转换不会报运行时错误。既然是顺序存储,那么适当合理地安排不同类型属性的出现顺序,可以达到优化内存使用的目的。比如将几个可以计算出占用内存和为8、16、32、64的属性放在一起,当然内存对齐的位数因不同平台和编译器而异。

类型定义通常还用在定义函数类型上,示例如下:

type Handler func(name string)

func process(h Handler) {
h("test")
}

类型别名

在go语言中,类型别名的定义示例如下:

package main

import "fmt"

func main() {
h := House{height: 1.0, width: 2.0}
fz := Fangzi{height: 1.0, width: 2.0}
fmt.Println(h.Area()) //House也有了Area方法
fmt.Println(fz.Area())
} type House struct {
height float32
width float32
} func (h *House) Height() float32 {
return h.height
} func (h *House) Width() float32 {
return h.width
} type Fangzi = House //类型别名 func (f *Fangzi) Area() float32 {
return f.height * f.width
}

  在语法上,类型别名与类型定义的差别仅仅是一个=号。但从上述示例可知,对Fangzi的改变,同样会体现在House上。Fangzi添加了Area方法,同时House也获得了计算Area的能力。这一点与类型定义是明显不同的。

  从这里可以看出,类型别名的一个好处在于可以为不喜欢的类型取一个漂亮的名字。比如,如果你更喜欢Human多于Person的话。

type Person struct {
name string
age int
sex string
} type Human = Person

  除此之外呢?对于库开发者来说,新旧版本包的迁移应是比较常见的事,特别是大版本升级的时候。那么为了保证新旧之间的兼容,可能就会用到类型别名的特性。比如旧包中的com.eventer/lib/AI,大版本升级时,需要变成com.eventer/lib2/AI。如果直接去掉lib/AI,则会造成使用这个库的项目,在引用新包时需要修改所有引用了旧包AI类型的代码。因而更好的作法是在移除com.eventer/lib/AI原有代码之后,加上一个类型别名的定义,即可完美解决这个问题。

package com.eventer/lib

import "com.eventer/lib2"

type AI = lib2.AI

  还有吗?如果下面的示例也能被称作一种用途的话,那么这一种用法就是"导出未被导出的类型"

type house struct {
name string
height float32
width float32
} func (h *house) getName() string {
return h.name
} func (h *house) GetArea() string {
return h.height * h.width
} type House = house

  但我认为这还是跟前文所述的类型别名的标准用法一般无异,实在看不出有什么不同,除了单词与原类型相同而已。否则根本没必要这样写代码,直接把house声明为House不是更直截了当?除非是用于取别名。

  如果go有泛型,那么类型别名在简化代码,提高代码可读性方面,无疑将带来极大的好处。特别是对于java语言那种泛型的复杂度让人抓狂的语言,只可惜java没有语言层面的类型定义语法。虽然可以通过一些取巧的方式也能达到类似的效果,比如继承。比如下面的例子:

public class LongArrayList extends ArrayList<Long> {

}

  上述代码定义一个Long类型的动态数组,相当于为泛型的ArrayList定义了一个新类型,在代码中可以直接使用。这是采用代码技巧突破在java中不能在顶级(top level)定义类型别名的限制,但 通过定义一个类来完成这件事情,是不是太重量级了。再比如:

class Test<I extends Integer> {
<L extends Long> void x(I i, L l) {
System.out.println(i.intValue() + ", " + l.longValue());
}
}

  这是在类级别或方法级别定义类型别名。上述例子是通过给Integer、Long定义更短的名字——I和L。

  反正这两种都只是语法形式上的变通,通常没人这样用。

  而在C语言中,有同样的功能typedef关键字,比如:

typedef struct Name1 {
elemtype ElemName;
} Name2, Name3;

其中Name1为结构体名,同时它还有两个"外号":Name2,Name3,因为c没有对象的概念,所以很明显这里的typedef并没有go的type关键字的内容多,但他们底层的原理应是相同的,都是通过在在编译时期替换完成。

请关注公众号

不一样的go语言-不同的语法之type的更多相关文章

  1. C语言-01-基本语法

    一.学前需知 开发工具 windows平台:Visual C++6.0等 mac平台:Xcode6.0等 以下文章内容皆是以Xcode6.0为开发工具,clang编译器. Xcode的一些常用快捷键 ...

  2. 关于JS脚本语言的基础语法

    JS脚本语言的基础语法:输出语法  alert("警告!");  confirm("确定吗?");   prompt("请输入密码");为弱 ...

  3. Xamarin XAML语言教程基础语法篇大学霸

    Xamarin XAML语言教程基础语法篇大学霸 前  言 Xamarin是一个跨平台开发框架.它可以用来开发iOS.Android.Windows Phone和Mac的应用程序.使用Xamarin框 ...

  4. Cocos2d-x 脚本语言Lua基本语法

    Cocos2d-x 脚本语言Lua基本语法 前面一篇博客对Lua这门小巧的语言进行了简单的介绍.本篇博客来给大家略微讲一下Lua的语法.不会长篇累牍得把Lua的全部语法都讲一遍,这里通过下面几点来讲L ...

  5. GO语言的基本语法之变量,常量,条件语句,循环语句

    GO语言的基本语法之变量,常量,条件语句,循环语句 作为慕课网得笔记自己看 定义变量: 使用var关键字 var a, b, C bool var s1, s2 string = "hell ...

  6. tn文本分析语言(二) 基本语法

    tn是desert和tan共同开发的一种用于匹配,转写和抽取文本的语言.解释器使用Python实现,代码不超过1000行. 本文主要介绍tn的基本语法.高级内容可以参考其他篇章.使用这样的语法,是为了 ...

  7. (转)JavaScript二:JavaScript语言的基本语法要求

    摘自:http://blog.csdn.net/erlian1992 要学习好JavaScript,首先我们要懂JavaScript语言的一些基本语法要求: 一,区分大小写 JavaScript语言区 ...

  8. javaweb学习总结七(XML语言作用、语法)

    一:XML语言的概念以及作用 1:xml概念:extensible Markup language,可扩展行标记语言,因为html的语法比较混乱,不够严谨. 用html写的系统不好维护,所以w3c组织 ...

  9. C语言-01基础语法

    1)         总结常见文件的拓展名 .c 是C语言源文件,在编写代码的时候创建 .o 是目标文件,在编译成功的时候产生 .out 是可执行文件,在链接成功的时候产生 2)         总结 ...

随机推荐

  1. Spring使用RMI进行远程方法调用

    (1).我新建了三个项目,SpringRmiApi(存放提供者和消费者共有的xx,例如实体类以及服务接口等等).SpringRmiService(服务提供者).SpringRmiProvider(服务 ...

  2. OpenCV:Debug和Release模式 && 静态和动态编译

    1.Release和Debug的区别 Release版称为发行版,Debug版称为调试版. Debug中可以单步执行.跟踪等功能,但生成的可执行文件比较大,代码运行速度较慢.Release版运行速度较 ...

  3. 本体【Ontology】综述

    原文地址:http://blog.csdn.net/moonsheep_liu/article/details/22329873 本体作为一种能在语义和知识层次上描述领域概念的建模工具,其目标是捕获相 ...

  4. Linux中涉及到计算优先级及其他问题

    比如计算矩形周长: a= b= echo `expr \* $((a+b))` 1.expr外要使用反引号,且expr只支持整数计算,如果涉及到浮点数计算要采用下面方法 2.优先计算a+b时,要使用双 ...

  5. vim常用

    删除空行 :g@^$@d

  6. js实现星级评分效果(非常规5个li代码)

    1. 前言 此方案受到JS单行写一个评级组件启发,自己写了一个简单Demo. 功能有正常滑动,动态显示实心星星个数:当点击确认,则保持当前的实心星星个数:再移动时未点击,则离开后还是保持之前的状态. ...

  7. win7 X64系统上 PL/SQL不能识别Oracle实例

    电脑系统为Win7 64位,安装的PLSql为64位,安装的Oracle客户端为运行时类型的,应该为32位客户端 电脑上之前安装的32位toad可以识别Oracle实例 在系统添加了oracle_ho ...

  8. 如何从现有版本升级到element UI2.0?使用npm-check-updates

    转:https://blog.csdn.net/wojiaomaxiaoqi/article/details/78428738 登录element UI官网时提示2.0已经正式发布了,Element ...

  9. OCM_第十一天课程:Section5 —》数据仓库

    注:本文为原著(其内容来自 腾科教育培训课堂).阅读本文注意事项如下: 1:所有文章的转载请标注本文出处. 2:本文非本人不得用于商业用途.违者将承当相应法律责任. 3:该系列文章目录列表: 一:&l ...

  10. 使用mybatis-spring-boot-starter如何打印sql语句

    只需要将接口文件的日志设置为debug即可. 例如你的mapper接口所在的文件夹是 com.demo.mapper 那么在application.properties配置文件中添加 logging. ...