超强总结:Go 语言四种类型转换方法

292次阅读  |  发布于2年以前

Go 语言是静态语言,在编写代码时,就对类型有严格的要求,一旦类型不匹配,就有可能编译失败。

因此在写代码时,就要经常用到类型的转换,这些知识点,对于一个新手来说,可以说是一个不小的门槛,经常要通常搜索引擎来寻找答案。

今天我总结了四种日常开发中最经常遇到的类型转换方法,吃透后就可以自由的进行类型转换。

# 第一种:显式类型转换

使用对应的类型函数进行转换,以一个简单的例子来帮助理解,使用 int16 就可以将 int8 类型的对象转为 int16

package main

import "fmt"

func main() {
    var a int8 = 5
    fmt.Printf("%T \n", a)
    // output: int8

    b := int16(a)
    fmt.Printf("%T \n", b)
    // output: int16
}

再举个例子,先将 string 类型通过 []byte 类型函数转为 []byte (等同于 []uint8),最后又使用 string 类型函数将 []byte 转回 string

package main

import "fmt"

func main() {
    var s1 string = "golang"
    fmt.Printf("%T \n", s1)
    // output: string

    s2 := []byte(s1)
    fmt.Printf("%T \n", s2)
    // output: []uint8

    s3 := string(s2)
    fmt.Printf("%T \n", s3)
    // output: string
}

这种方法,同样适用于自己定义的结构体和接口类型,但要注意的是,仅能用于将结构体类型转换接口类型,而不能将接口类型转为结构体类型。

这边也举个例子

package main

import "fmt"

type People interface {
    Speak()
}

type Student struct {
    name string
}

func (s Student) Speak() {
    fmt.Println("hello, golang")
}

func demo2(s People) {
    s.Speak()
}

func demo1(s1 Student) {
        // 结构体类型转为接口类型
    s2 := People(s1)
    demo2(s2)
}

func main() {
    s1 := Student{name: "wangbm"}
    demo1(s1)
}

# 第二种:隐式类型转换

隐式转换,是编译器所为,在日常开发中,开发者并不会感觉到发生了变化。

隐式转换以下面两种情况最为常见,非常简单,我直接以截图加标注说明就行了,不再长篇大论。

函数调用时转换

函数返回时转换

# 第三种:类型断言

学了上面第二种方法后,我们已经知道下面这段代码,肯定是会编译失败的。

package main

import "fmt"

type Student struct {
    name string
}

func (s Student) Speak() {
    fmt.Println("hello, golang")
}

func demo2(s Student) {
    s.Speak()
}

func demo1(s1 interface{}) {
    demo2(s1)
}

func main() {
    s1 := Student{name: "wangbm"}
    demo1(s1)
}

答案当然是不能了,原因很简单,因为经过 demo1 函数后,s1 会被隐式转换成 interface{} 类型,而 demo2 函数的参数类型要求为 Student,因此类型不匹配。

解决方法也相当的简单,只要使用类型断言一下,就能实现静态类型的转换。

为了避免有新手,还不了解类型断言(Type Assertion),我这边再简单介绍一下。

类型断言可用于判断一个对象是否是某类型。

这其中包含两种情况:

第一种情况:该对象是 T 类型(struct 类型),则断言该对象是 T 类型,就能断言成功。

第二种情况:该对象是 I 类型(接口类型),则断言对象是 T 类型,也能断言成功,并且返回一个静态类型为 T 的对象,也相当于类型转换了。

倘若类型断言失败,则会抛出 panic,使用的时候,请千万注意处理。若不想让其抛出 panic,可以使用另一种断言语法。断言不是今天的主题,这里不再展开,更多具体内容,请查看我以前的这篇文章 ([27. 说说 Go 语言中的类型断言?] ),写的非常清楚。

s, ok := x.(T)

另外,有一点需要提醒的是,类型断言并不能用于两个通用类型的相互转换,只能用于将静态类型为 interface{} 类型的对象,转为具体的类型。

# 第四种:重新构造对象

在之前的教程中([34 图解:静态类型与动态类型] ),我使用图解,详细的解释了 Go 语言中的静态类型与动态类型。

其中有一个非常重要的知识点,就是如下这种定义变量的方法

package main

import "fmt"

func main() {
    age := (int)(25)
    // 等价于 age := 25
    fmt.Printf("type: %T, data: %v ", age, age)
    // output: type: int, data: 25
}

思路变通一下,这个知识点,也可以应用于类型的转换。

package main

import "fmt"

func main() {
    var a int8 = 5
    fmt.Printf("%T \n", a)
    // output: int8

    b := (int16)(a)
    fmt.Printf("%T \n", b)
    // output: int16
}

以上,就是我总结的四种类型转换方法,如果你有更多的原生类型转换方法,欢迎后台留言告诉我。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8