go之数据类型转换和类型断言

一、类型转换

1.1 简单类型转换

  • 格式
valueOfTypeB = typeB(valueOfTypeA)
  • int 转 float64
package main

import "fmt"

func main(){
	var num int = 12
	var f float64
	f = float64(num)
	fmt.Println(f)
}

  • float64 转int
package main

import "fmt"

// float64 -> int
func main(){
	var f float64 = 12.4341
	var num int
	num = int(f)
	fmt.Println(num)
}
  • float64 转 uint8
package main

import "fmt"

// float64 -> int
func main(){
	var f float64 = 257.4341
	var num uint8
	num = uint8(f)
	fmt.Println(num)
}

结果:
    1
为什么不是257?
 因为uint8的数字范围是 0 ~255,而257 明显超过了0~255,所以go会重新从0开始,此时257 也就是 1
  • 注意
1、不是所有的数据类型都可以转换的,例如string 转 int 就不可以
2、低精度转化成高精度是安全的,但是高精度转低精度就会丢失精度
3、简单的类型转换方式不能对int(float) 和 string 互转,需要通过strconv 包实现

1.2 strconv

  • strconv 包提供了简单的数据类型之间的类型转换功能
  • int -> string
var num int = 12
var s string
s = strconv.Itoa(num)
  • int -> string
var str string = "34"
var n int
n,err := strconv.Atoi(str) // string 类型的数据并不一定是整型数据。
if err != nil {
    fmt.Println("str 不是合法的int型的数据")
}
  • string -> bool
  • string -> int
  • string -> float
  • string -> uint
package main

import (
	"fmt"
	"strconv"
)

func main(){
	var (
		str1 = "true"
		str2 = "3.1415"
		str3 = "-42"
		str4 = "256"
	)

	var b bool

	b,err := strconv.ParseBool(str1)
	if err != nil{
		fmt.Println("str1 不是合法的 bool型的数据")
	}
	fmt.Println(b)

	var f float64
	f,err = strconv.ParseFloat(str2,64) // 64 表示64位
	if err != nil{
		fmt.Println("str2 不是合法的 float64型的数据")
	}
	fmt.Println(f)

	var i int64
	i ,err = strconv.ParseInt(str3,10,64) // 第二个参数10 表示是多少进制的,第三位是int64类型的数据
	if err != nil{
		fmt.Println("str3 不是合法的 int64型的数据")
	}
	fmt.Println(i)

	var u uint64
	u,err = strconv.ParseUint(str4,10,8)
	if err !=nil {
		fmt.Println("str4 不是合法的 uin8型的数据")
	}
	fmt.Println(u)
}
结果:
true
3.1415
-42
str4 不是合法的 uin8型的数据
255
  • float -> string
  • int -> string
  • uint -> string
  • bool -> string
package main

import (
	"fmt"
	"strconv"
)

func main() {
	var (
		b bool    = true
		f float64 = 3.141555
		i int64   = -42
		u uint64  = 24
	)

	// bool -> string
	var str1 string
	str1 = strconv.FormatBool(b)
	fmt.Println(str1)

	// float -> string
	var str2 string
	str2 = strconv.FormatFloat(f, 'f', -1, 64) // 第二个参数是格式,
	fmt.Println(str2)

	// int -> string
	var str3 string
	str3 = strconv.FormatInt(i, 10) // 第二个参数10 表示是多少进制的
	fmt.Println(str3)

	// uint ->string
	var str4 string
	str4 = strconv.FormatUint(u, 10)
	fmt.Println(str4)
}

结果:
true
3.1415
-42
24

类型断言

  • 什么是类型断言
断言某种数据的类型
  • 为什么要用类型断言
在go语言中接口值的动态类型是不确定的
  • 如何使用类型断言
变量,ok = 空接口变量.(数据类型)
  • 简单运用
package main

import "fmt"

func main(){
	var num interface{}
	num = 12
	i,ok := num.(int)
	if !ok{
		fmt.Println("num 不是int类型的数据")
	}
	fmt.Println(i)

	var str interface{}
	str = "jmz"
	u,ok := str.(uint)
	if !ok {
		fmt.Println("str 不是uint类型的数据")
	}
	fmt.Println(u)
}
结果:
12
str 不是uint类型的数据
0
  • 为什么下面的转换就不行
package main

import "fmt"

type Enum int
func main(){
	var n Enum = 12
	var i interface{}
	i = n
	num ,ok := i.(int)
	if !ok {
		fmt.Println("i 不是 int 类型")
	}
	fmt.Println(num)
}

结果
i 不是 int 类型
0
疑问:
 12 明明可以是int 类型,为什么不能断言成int类型?
 上面说接口值的动态类型是不确定的,应该可以转换成int呀?
解惑:
    1、golang 中的变量包含有两个部分(type,value),变量的类型和值
    2、每一个interface变量对应的是一个pair,pair 记录的是实际的类型和值

例子:
    var num interface{} = 12  // 该变量 对应的pair是(nil,12)

    var str string = "jmz"
    var s interface{} = str   // 该变量对应的pair 是(string,"jmz")
因为:
    var n Enum = 12
    var i interface{}
    i = n   // i 接口变量指向的pair的实际类型和值是(Enum,12)
    
所以:
    num ,ok := i.(int)   // i是不能断言成int 的,因为i 的动态类型是明确的 Enum
  • 注意
    1、在golang 中一种特殊类型:interface{} 类型,所有类型都默认继承了interface{} 类型
    2、golang 中的变量包含有两个部分(type,value) 即 变量的类型和变量的值
    3、每一个interface 变量都有一个对应的pair ,pair 记录了实际的变量的值和类型(type,value)
    4、interface{}类型的变量包含了2个指针,一个指针指向值的类型,一个指针指向实际的值

如果喜欢看小说,请到183小说网

posted @ 2019-04-07 22:42  xiaobaiskill  阅读(992)  评论(0编辑  收藏  举报