您好,欢迎访问一九零五行业门户网

Go语言类型别名是什么

在go语言中,类型别名就是为已存在的“类型”定义一个别名,可以使用type关键字来定义,语法“type typealias = type”。类型别名是go 1.9版本添加的新功能,主要用于解决代码升级、迁移中存在的类型兼容性问题;在 c/c++ 语言中,代码重构升级可以使用宏快速定义一段新的代码,go语言中没有选择加入宏,而是解决了重构中最麻烦的类型名变更问题。
本教程操作环境:windows7系统、go 1.18版本、dell g3电脑。
go语言 类型别名是什么
golang 中类型别名就是为已存在的 类型 定义一个别名。golang 中类型别名使用 type 关键字来定义。
语法
type typealias = type

参数
参数描述
type 定义类型别名使用的关键字。
typealias type 的别名。
type 需要起别名的类型。
类型别名是 go 1.9 版本添加的新功能,主要用于解决代码升级、迁移中存在的类型兼容性问题。在 c/c++ 语言中,代码重构升级可以使用宏快速定义一段新的代码,go语言中没有选择加入宏,而是解决了重构中最麻烦的类型名变更问题。
在 go 1.9 版本之前定义内建类型的代码是这样写的:
type byte uint8type rune int32
而在 go 1.9 版本之后变为:
type byte = uint8type rune = int32
这个修改就是配合类型别名而进行的修改。
区分类型别名与类型定义
定义类型别名的写法为:
type typealias = type

类型别名规定:typealias 只是 type 的别名,本质上 typealias 与 type 是同一个类型,就像一个孩子小时候有小名、乳名,上学后用学名,英语老师又会给他起英文名,但这些名字都指的是他本人。
类型别名与类型定义表面上看只有一个等号的差异,那么它们之间实际的区别有哪些呢?下面通过一段代码来理解。
package mainimport ( "fmt")// 将newint定义为int类型type newint int// 将int取一个别名叫intaliastype intalias = intfunc main() { // 将a声明为newint类型 var a newint // 查看a的类型名 fmt.printf("a type: %t\n", a) // 将a2声明为intalias类型 var a2 intalias // 查看a2的类型名 fmt.printf("a2 type: %t\n", a2)}
代码运行结果:
a type: main.newinta2 type: int
代码说明如下:
第 8 行,将 newint 定义为 int 类型,这是常见的定义类型的方法,通过 type 关键字的定义,newint 会形成一种新的类型,newint 本身依然具备 int 类型的特性。
第 11 行,将 intalias 设置为 int 的一个别名,使用 intalias 与 int 等效。
第 16 行,将 a 声明为 newint 类型,此时若打印,则 a 的值为 0。
第 18 行,使用%t格式化参数,打印变量 a 本身的类型。
第 21 行,将 a2 声明为 intalias 类型,此时打印 a2 的值为 0。
第 23 行,打印 a2 变量的类型。
结果显示 a 的类型是 main.newint,表示 main 包下定义的 newint 类型,a2 类型是 int,intalias 类型只会在代码中存在,编译完成时,不会有 intalias 类型。
非本地类型不能定义方法
能够随意地为各种类型起名字,是否意味着可以在自己包里为这些类型任意添加方法呢?参见下面的代码演示:
package mainimport ( "time")// 定义time.duration的别名为mydurationtype myduration = time.duration// 为myduration添加一个函数func (m myduration) easyset(a string) {}func main() {}
代码说明如下:
第 8 行,为 time.duration 设定一个类型别名叫 myduration。
第 11 行,为这个别名添加一个方法。
编译上面代码报错,信息如下:
cannot define new methods on non-local type time.duration
编译器提示:不能在一个非本地的类型 time.duration 上定义新方法,非本地类型指的就是 time.duration 不是在 main 包中定义的,而是在 time 包中定义的,与 main 包不在同一个包中,因此不能为不在一个包中的类型定义方法。
解决这个问题有下面两种方法:
将第 8 行修改为 type myduration time.duration,也就是将 myduration 从别名改为类型;
将 myduration 的别名定义放在 time 包中。
在结构体成员嵌入时使用别名
当类型别名作为结构体嵌入的成员时会发生什么情况呢?请参考下面的代码。
package mainimport ( "fmt" "reflect")// 定义商标结构type brand struct {}// 为商标结构添加show()方法func (t brand) show() {}// 为brand定义一个别名fakebrandtype fakebrand = brand// 定义车辆结构type vehicle struct { // 嵌入两个结构 fakebrand brand}func main() { // 声明变量a为车辆类型 var a vehicle // 指定调用fakebrand的show a.fakebrand.show() // 取a的类型反射对象 ta := reflect.typeof(a) // 遍历a的所有成员 for i := 0; i < ta.numfield(); i++ { // a的成员信息 f := ta.field(i) // 打印成员的字段名和类型 fmt.printf("fieldname: %v, fieldtype: %v\n", f.name, f.type. name()) }}
代码输出如下:
fieldname: fakebrand, fieldtype: brandfieldname: brand, fieldtype: brand
代码说明如下:
第 9 行,定义商标结构。
第 13 行,为商标结构添加 show() 方法。
第 17 行,为 brand 定义一个别名 fakebrand。
第 20~25 行,定义车辆结构 vehicle,嵌入 fakebrand 和 brand 结构。
第 30 行,将 vechicle 实例化为 a。
第 33 行,显式调用 vehicle 中 fakebrand 的 show() 方法。
第 36 行,使用反射取变量 a 的反射类型对象,以查看其成员类型。
第 39~42 行,遍历 a 的结构体成员。
第 45 行,打印 vehicle 类型所有成员的信息。
这个例子中,fakebrand 是 brand 的一个别名,在 vehicle 中嵌入 fakebrand 和 brand 并不意味着嵌入两个 brand,fakebrand 的类型会以名字的方式保留在 vehicle 的成员中。
如果尝试将第 33 行改为:
a.show()
编译器将发生报错:
ambiguous selector a.show
在调用 show() 方法时,因为两个类型都有 show() 方法,会发生歧义,证明 fakebrand 的本质确实是 brand 类型。
【相关推荐:go视频教程、编程教学】
以上就是go语言类型别名是什么的详细内容。
其它类似信息

推荐信息