4.4.4-3 基础类型

Go 语言内置的基础类型包括:

  • 布尔类型

  • 数字类型,例如整数、浮点数、复数等

  • 字符串类型

编译器使用同一个结构来表示所有的基础类型,其定义如下:

// A Basic represents a basic type.
type Basic struct {
    kind BasicKind
    info BasicInfo
    name string
}

BasicInfo用来描述类型属性,例如该类型是否是数字类型,是否可排序。其定义如下:

type BasicInfo int

// Properties of basic types.
const (
    IsBoolean BasicInfo = 1 << iota
    IsInteger
    IsUnsigned
    IsFloat
    IsComplex
    IsString
    IsUntyped

    IsOrdered   = IsInteger | IsFloat | IsString
    IsNumeric   = IsInteger | IsFloat | IsComplex
    IsConstType = IsBoolean | IsNumeric | IsString
)

BasicInfo用来标识类型的性质,对应的常量申明已经解释了其潜在用途。

BasicKind用来标识实际类别,例如Int32, Float64等;这里简单罗列几个以做示例:

可以看出对于每一个基础类型,都有一个枚举类型与之对应,另外可以发现 go 中的byterune类型只是uint8int32的类型别名。

这里稍稍多讲一下 Untypedxxx 这一组类型。go 的类型系统非常严格,如果一个表达式的类型是确定的,那么不管在什么场景下,编译器都不会做任何隐式的类型转换,例如如下代码:

在赋值语句c := a + b 中,a与b的类型都是确定的整数类型,而将 a 转换为 int32 也是安全的,但 go 并不会这么做。反观 Java 则比较灵活,其定义了一组类型的隐式转换规则,例如如下 Java 代码是合法的:

但是 go 也没有死板到无可救药,对于类型还不确定的表达式,go 会根据上下文来进行类型转换。但 go 中仅允许常量在声明时拥有不确定的类型,对应BasicKindUntypedxxx的几类。

更准确地说,go 基础类型的字面量都是 untyped 的,例如 "golang", 12, 3.14 分别对应 UntypedString, UntypedInt, UntypedFloat. 当使用这些字面量进行操作,例如赋值、运算、作为参数传递时,编译器会根据上下文将其转换为目标类型。我们来看几个示例:

  • 常量声明有类型

"Golang" 类型为 UntypedString, 但是 name 的声明中指定类型为 string, 所以编译器在初始化 name 时会将 "Golang" 转换为 string 类型并且赋值给 name. 如果此时类型转换不匹配,则会报错:

  • 常量声明无类型

name 没有申明类型,所以其类型复用字面量 "Golang" 的类型 UntypedString; 而 alias 的类型是 string, name 的值赋给 alias 的时候编译器会将其转换为 string 类型

  • 根据常量 underlying type 进行转换

这里我们声明了新的类型 Str, 其 underlying type 是 string, 通过对 alias 的赋值语句可以看出:编译器依然可以将 UntypedString 转换为 Str 类型。但 owner 被显示声明成了 string 类型,其与 Str 毕竟是不同的类型,所以 ownerAlias 的赋值语句中,编译器不会将 string 类型的 owner 隐式转换为 Str 类型。

  • 常量声明无类型

上面我们提到 go 仅仅支持常量拥有不确定类型,所以即使变量 name 没有申明类型,但编译器会根据右边的值推测出一个具体的类型赋给 name, 这里 "Golang" 的类型是 UntypedString。UntypedString 只能被转换为 string 类型,所以对 alias 的赋值会报错。

Untyped 类型的转换规则很简单:数字类型低精度可以向高精度转换,除此之外只有相同的类型才能相互转换。

最后更新于

这有帮助吗?