01 介绍
Go 语言是强类型编程语言,一些使用弱类型编程语言的读者朋友们在初学 Go 语言时,多多少少都会不太适应 Go 语言的类型。
Go 语言变量类型包含基础类型和复合类型,类型转换和类型断言一般是对基础类型的处理,基础类型包含整数、浮点数、布尔和字符串。
其中整数类型又称为整型,分为有符合和无符号,各自又包含不同大小,8位、16位、32位和64位,其中 int32 和 uint8 的类型别名分别是 rune和 byte。
浮点数类型分为 float32 和 float64,为了避免精度丢失,一般我们选择使用 float64,float32 和 float64 之间可以直接转换,整型和浮点数类型之间也可以直接转换,需要注意丢失精度的问题。
布尔类型的值只有两个,分别是 true 和 false,类型零值为 false。需要注意的是它无法像弱类型编程语言可以隐式转换为 1 和 0。
字符串类型是一组使用双引号引起来的字节序列,它可以包含任意数据。需要注意的是它不可以改变,因为多个字符串可以共享同一块内存空间。
本文我们介绍 Go 语言的类型转换和类型断言。
02 类型转换
我们在项目开发时,可能会遇到一些需要类型转换的场景,比如我们使用 Go 语言开发 Api 接口。
客户端(调用方)在请求我们使用 Go 语言开发的 Api 接口时,虽然会按照我们预先协商的参数类型,但是随着项目的不断迭代,可能以前定义的变量类型需要修改。
因为 Go 语言是强类型语言,不支持类型隐式转换,我们就需要显式转换变量的类型。
Go 语言类型转换的方式:
强制转换
整数类型之间可以强制转换,代码如下:
func main(){
var a int64
a = 1
fmt.Printf("%T\t%d\n", a, a)
var b int8
b = int8(a)
fmt.Printf("%T\t%d\n", b, b)
}
阅读上面这段代码,我们定义 int64 类型的变量 a,使用 <类型>(<数值>) 的格式,直接把变量 a 的由 int64 转换为 int8 的变量 b。
浮点数类型之间,浮点数和整型之间,也可以强制转换,代码如下:
func main(){
var a float64
a = 3.1415926
fmt.Printf("%T\t%f\n", a, a)
var b float32
b = float32(a)
fmt.Printf("%T\t%f\n", b, b)
var c int64
c = int64(b)
fmt.Printf("%T\t%d\n", c, c)
}
阅读上面这段代码,我们定义 float64 类型的变量 a,使用 <类型>(<数值>) 的格式,直接把变量 a 由 float64 转换为 float32 的变量 b,然后变量 b 由 float32 转换为 int64 的变量 c。需要注意丢失精度的问题。
布尔类型 bool,它的值只有两个,分别是 true 和 false,它没有其它类型可以强制转换,不过可以使用标准库或三方库对布尔类型进行类型转换。
字符串类型是一组使用双引号引起来的字节序列,所以 string 和 []byte 之间可以强制转换,代码如下:
func main(){
var a string
a = "golang"
fmt.Printf("%T\t%s\n", a, a)
var b []byte
b = []byte(a)
fmt.Printf("%T\t%d\n", b, b)
}
阅读上面这段代码,我们定义 string 类型的变量 a,使用 <类型>(<数值>) 的格式,直接把变量 a 由 string 转换为 []byte 的变量 b,反之亦然。
使用标准库或三方库
无法强制转换的类型,可以使用标准库或三方库,比如布尔类型,代码如下:
func main(){
var a bool
a = true
fmt.Printf("%T\t%t\n", a, a)
var b string
b = strconv.FormatBool(a)
fmt.Printf("%T\t%s\n", b, b)
}
阅读上面这段代码,我们定义 bool 类型的变量 a,使用 <类型>(<数值>) 的格式,使用标准库 strconv 的方法把变量 a 由 bool 转换为 string 的变量 b。
除了标准库 strconv[1] 之外,标准库 fmt[2] 也提供了类型转换的方法;还有一些三方库,比如 cast[3]。限于篇幅,此处不再详细赘述,感兴趣的读者朋友们可以阅读相关文档了解更多。
03 类型断言
我们在项目开发时,可能想要定义参数的类型为通用类型,比如我们使用 Go 语言开发 Api 接口。
我们想要尽量适配客户端(调用方)传参使用不同类型,比如调用方是使用弱类型编程语言的场景。
我们可以定义变量类型的空接口类型 interface{},然后使用类型断言,获取传参的实际类型,按需处理为我们想要的类型。
示例代码:
func main(){
var id interface{}
id = 1 // 参数 id 接收到的值为整型
fmt.Printf("%T\t%v\n", id, id)
// 需要使用字符串类型的变量 id 赋值给字符串类型的变量 uid
var uid string
value, ok := id.(string)
if ok {
uid = value
}
fmt.Printf("%T\t%v\n", uid, uid)
}
阅读上面这段代码,我们定义 interface{} 空接口类型的变量 id,作为接收请求参数,实际需要使用字符串类型的数据,我们使用类型断言检查变量 id 的值是否是字符串类型,是字符串类型则赋值给变量 uid。
需要注意的是,我们在使用类型断言时,最好使用 ok-idiom 模式,避免引发 panic。
此外,还有 switch case 方式的类型断言,也称为类型选择。可以处理多种类型,代码如下:
func main() {
var id interface{}
id = 1 // 参数 id 接收到的值为整型
fmt.Printf("0-%T\t%v\n", id, id)
// 需要使用字符串类型的变量 id 赋值给字符串类型的变量 uid
var uid string
switch val := id.(type) {
case string:
uid = val
fmt.Printf("1-%T\t%v\n", uid, uid)
case int:
uid = strconv.Itoa(val)
fmt.Printf("2-%T\t%v\n", uid, uid)
default:
fmt.Printf("3-%T\t%v\n", uid, uid)
}
}
阅读上面这段代码,我们使用 switch case 方式的类型断言参数 id,如果参数的值是我们需要的类型,则直接使用,反之,则类型转换之后再使用。
细心的读者朋友们可能发现该方式的类型断言格式有所不同,小括号中的数据类型改为 type。
需要注意的是,使用 switch case 方式的类型断言,即便省略 default,也不会因为不是 ok-idiom 模式的类型断言而引发 panic。
04 总结
本文我们介绍 Go 语言中让之前一直使用弱类型编程语言的读者朋友们迷惑的类型转换和类型断言。
读完本文,大家至少可以区分类型转换和类型断言的区别,和了解各自的使用场景。