zoukankan      html  css  js  c++  java
  • [Golang学习笔记] 06 程序实体3 类型断言和类型转换

    类型断言:

    语法:
    <目标类型的值>,<布尔参数> := <表达式>.( 目标类型 ) // 安全类型断言
    <目标类型的值> := <表达式>.( 目标类型 )  //非安全类型断言

    x.(T),这里x表示一个接口的类型,T表示一个类型(也可为接口类型)。
    一个类型断言检查一个接口对象x的动态类型是否和断言的类型T匹配。

    类型断言分两种情况:
    第一种,如果断言的类型T是一个具体类型,类型断言x.(T)就检查x的动态类型是否和T的类型相同。

    如果这个检查成功了,类型断言的结果是一个类型为T的对象,该对象的值为接口变量x的动态值。换句话说,具体类型的类型断言从它的操作对象中获得具体的值。
    如果检查失败,接下来这个操作会抛出panic,除非用两个变量来接收检查结果,如:f, ok := w.(*os.File)
    第二种,如果断言的类型T是一个接口类型,类型断言x.(T)检查x的动态类型是否满足T接口。

    如果这个检查成功,则检查结果的接口值的动态类型和动态值不变,但是该接口值的类型被转换为接口类型T。换句话说,对一个接口类型的类型断言改变了类型的表述方式,改变了可以获取的方法集合(通常更大),但是它保护了接口值内部的动态类型和值的部分。
    如果检查失败,接下来这个操作会抛出panic,除非用两个变量来接收检查结果,如:f, ok := w.(io.ReadWriter)
    注意:

    如果断言的操作对象x是一个nil接口值,那么不论被断言的类型T是什么这个类型断言都会失败。
    我们几乎不需要对一个更少限制性的接口类型(更少的方法集合)做断言,因为它表现的就像赋值操作一样,除了对于nil接口值的情况。

    面试题:怎么样判断一个变量的类型?

    package main
    
    import "fmt"
    
    var container = []string{"zero", "one", "two"}
    
    func main() {
        container := map[int]string{0: "zero", 1: "one", 2: "two"}
        fmt.Printf("The element is %q.
    ", container[1])
    }

    回答:

    使用“类型断言”表达式。
    value, ok := interface{}(container).([]string)
    在赋值符号的右边,是一个类型断言表达式。
    把container变量的值转换为空接口值的interface{}(container)。
    以及一个用于判断前者类型是否为切片类型[]string的 .([]string)。

    解析:
    类型断言表达式的语法形式是x.(T)。其中x代表要被判定类型的值。这个值的类型必须是接口类型的,不过无关哪个接口类型。
    所以如果container不是任何的接口类型时,需要先转换成某个接口类型。
    如果是,这个断言表达是可以写成container.([]string)。

    最右边的圆括号中 []string 是一个类型字面量。类型字面量,就是用来表示数据类型本身的若干个字符。
    比如,string是表示字符类型的字面量,uint8是表示8位无符号整数类型的字面量。

    一对不包裹任何东西的花括号,既可以代表空代码块,也可以表示不包含任何内容的数据结构(数据类型)。

    struct{}表示不包含任何字段和方法的空结构体类型。
    []string{}表示空切片值(数据类型)。
    map[int]string{}表示空字典值,类型字面量是表示数据类型本身的若干个字符,如string是字符串类型的字面量,uint8表示8位无符号整数类型的字面量。
    其中[]string表示元素类型为string的切片类型;
    map[int]string表示键类型为int,值类型为string的字典类型。


    类型转换:

    语法:<结果类型> := <目标类型> ( <表达式> )
    T(x)
    T表示类型,x可以是变量,也可以是一个代表值的字面量(比如1.23和struct{})

    var a int = 9
    var b := float(a)

    类型转换是用来在不同但相互兼容的类型之间的相互转换的方式,所以,当类型不兼容的时候,是无法转换的。

    int<--->string

    //string到int
    value_int,err:=strconv.Atoi(string)
    //int到string
    str:=strconv.Itoa(value_int)

    int64<--->string

    //string到int64
    value_int64, err := strconv.ParseInt(string, 10, 64)
    //int64到string,需注意下面转换规定
    //FormatInt returns the string representation of i in the given base, for 2 <= base <= 36.
    //The result uses the lower-case letters 'a' to 'z' for digit values >= 10
    str:=strconv.FormatInt(value_int64, 10) //FormatInt第二个参数表示进制,10表示十进制

    float<--->string

    //float转string
    v := 3.1415926535
    s1 := strconv.FormatFloat(v, 'E', -1, 32)//float32 
    s2 := strconv.FormatFloat(v, 'E', -1, 64)//float64
    //第二个参数可选'f'/'e'/'E'等,含义如下:
    // 'b' (-ddddp±ddd,二进制指数)
    // 'e' (-d.dddde±dd,十进制指数)
    // 'E' (-d.ddddE±dd,十进制指数)
    // 'f' (-ddd.dddd,没有指数)
    // 'g' ('e':大指数,'f':其它情况)
    // 'G' ('E':大指数,'f':其它情况)
     
    //string转float
    s := "3.1415926535"
    v1, err := strconv.ParseFloat(v, 32)
    v2, err := strconv.ParseFloat(v, 64)

    float<--->int

    var a int64
    a = 1
    var b float64
    b = 2.000
     
    //a -- float64
    c := float64(a)
     
    //b -- int64
    d := int64(b)

    思考题:1. 你认为类型转换规则中有哪些值得注意的地方?

    1、 对于整数类型值、整数常量之间的类型转换,原则上只要源值在目标类型的可表示范围内就是合法的。
    源整数类型的可表示范围较大,而目标类型的可表示范围较小的情况,比如把int16转换为int8,如:

    var srcInt = int16(-255)
    dstInt := int8(srcInt)

    其中变量srcInt的值是int16类型的-255转化来的,而变量dstInt的值是srcInt转换来的,类型是int8。

    此例子中,int16类型比int8的范围要大,运行结果为:1.
    整数在Go中是以补码的形式存储的,补码其实就是原码个位求反再加1.
    int16类型的值-255的补码是1111111100000001,如果转换为int8类型,那么Go就会把在较高位置(最左位置)上的8位二进制直接截掉,从而得到00000001.
    又由于最左边一位是0,表示它是个正整数,而正整数的补码就等于其原码,所以这里输出1.

    2、虽然直接把一个整数值转为一个string类型的值是可行的,但是被转换的整数值必须要能代表一个有效的Unicode代码点。

    3、string类型与各种切片类型之间的互转。
    一个值从string类型向[]byte类型转换时,代表着以UTF-8编码的字符串会被拆分成独立的字节,两个字节代表一个汉字。
    一个值从string类型向[]rune类型转换时,代表着字符串会被拆分成一个个的Unicode字符,一个字符代表一个汉字。

    2. 什么是别名类型?什么是潜在类型?

    别名类型是type声明自定义类型的一种,如:

    type MyString = string

    表示MyString是string类型的别名类型,与其源类型是完全相同的。
    别名类型主要是为了代码重构而存在的。
    byte是uint8的别名类型,rune是int32的别名类型。

    潜在类型是某个类型在本质上是哪个类型或者是哪个类型的集合,如:

    type MyString2 string

    潜在类型的定义之间没有“=”号
    潜在类型的值之间是可以进行类型转换的,如本例中MyString2类型的值可以跟string类型的值互相转换,但是集合类的类型[]MyString2与[]string就不行,

    因为他们的潜在类型就不同了,分别是MyString2和string。

    即使两个类型的潜在类型相同,他们的值之间也不能进行判断或比较,他们的变量之间也不能互相赋值。

    3. 别名类型在代码重构过程中可以起到哪些作用吗?

    类型别名是 Go 1.9 版本添加的新功能。主要用于代码升级、迁移中类型的兼容性问题。
    下一篇分析类型别名和类型定义。

    总结
    类型断言表达式可用来判断变量是哪个类型,把结果赋给两个变量,要保证被判断的变量是接口类型的,这可能会用到类型转换表达式。
    要搞清楚别名类型声明与类型再定义之间的区别。

    本学习笔记仅为了总结自己学到的Go语言核心知识,方便以后回忆,文中部分内容摘录自极客时间的《Go语言核心36讲》专栏,如有侵权,请联系我删除。

     

  • 相关阅读:
    C# SocketAsyncEventArgs Server
    C# Socket 入门5 UPD 结构体 与 C++ 通信
    如何取得 Func 对象 字段
    动态调用对象的属性和方法——性能和灵活性兼备的方法
    C# 读写锁 ReaderWriteLock
    C# IP多播
    C# 广播通信
    程序员那点儿事
    wince上数据库:sqlce和sqlite
    evc vc字符串转换处理
  • 原文地址:https://www.cnblogs.com/fyql/p/10206426.html
Copyright © 2011-2022 走看看