go类型转换及与C的类型转换方式


Posted in Golang onMay 05, 2021

GO类型转换及与C的类型转换

类型转换

语法

dst := float32(src)

示例

var num int = 520
f32 := float32(num)
i64 := int64(num)

注意:加入val是一个指针,int32(val)相当于*(int32(var1)),指针不能直接转换成一个int32类型,应该改为(*int32)(var1),所以某些类型可能引起误会的应该用括号括起来转换。

类型断言

语法

dst,ok := src.(int) // 安全类型断言,通过ok判断是否转换成功
dst := src.(int)   // 非安全类型断言,无法确定是否转换成功

示例

var val interface{} = 123
num,ok := val.(int)
if ok {
    fmt.Printf("%T->%d\n", num, num)
} else {
    fmt.Println("类型不匹配")
}

其他转换

go提供了strconv包来进行转换

bytes 、string转换

//类型转换  string to bytes 
func str2bytes(s string) []byte { 
    x := (*[2]uintptr)(unsafe.Pointer(&s)) 
    h := [3]uintptr{x[0], x[1], x[1]} 
    return *(*[]byte)(unsafe.Pointer(&h)) 
}
//类型转换  bytes to string 
func bytes2str(b []byte) string { 
    return *(*string)(unsafe.Pointer(&b)) 
}

interface转为string

//interface转为string 
func interface2string(inter interface{}) string { 
    tempStr := "" 
    switch inter.(type) { 
        case string: tempStr = inter.(string) break 
        case float64: tempStr = strconv.FormatFloat(inter.(float64), 'f', -1, 64) break 
        case int64: tempStr = strconv.FormatInt(inter.(int64), 10) break 
        case int: tempStr = strconv.Itoa(inter.(int)) break 
 } 
    return tempStr 
}
//整形转字符串
strconv.Itoa(100)
//该方法的源码是:
//可以看出是FormatInt方法的简单实现。
func Itoa(i int) string {
        return FormatInt(int64(i), 10)
}
//字符串转整形
i, _ := strconv.Atoi("100")
//64位整形转字符串,FormatInt第二个参数表示进制,10表示十进制
var i int64
i = 0x100
strconv.FormatInt(i, 10)
//字节转32位整形
b := []byte{0x00, 0x00, 0x03, 0xe8}
bytesBuffer := bytes.NewBuffer(b)
//其中binary.BigEndian表示字节序,相应的还有little endian。通俗的说法叫大端、小端。
var x int32
binary.Read(bytesBuffer, binary.BigEndian, &x)
fmt.Println(x)
//32位整形转字节
var x int32
x = 106
bytesBuffer := bytes.NewBuffer([]byte{})
binary.Write(bytesBuffer, binary.BigEndian, x)
fmt.Println(bytesBuffer.Bytes())
//字节转字符串
fmt.Println(string([]byte{97, 98, 99, 100}))
//字符串转字节
fmt.Println([]byte("abcd"))

string与unicode字符互转

//string-->unicode
func Str2uft16ptr(str string)(p uintptr){//将字符串转为utf16指针
 e:=utf16.Encode([]rune(str))//转成unicode
 e=append(e, uint16(0))//添加末尾的0
 p=uintptr(unsafe.Pointer(&e[0]))//转成指针
 //p=uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(str)))
 return
}
//unicode-->string
func addbuf(buf []uint16,newcap int)(newbuf []uint16){
 newbuf=make([]uint16,newcap)
 copy(newbuf,buf)
 return
}
func Utf16prt2str(p uintptr)(str string){//将utf16指针转为字符串
 len:=0
 buf:=make([]uint16,64)
 for a:=(*(*uint16)(unsafe.Pointer(p)));a!=0;len++{
  if len>=cap(buf){
   buf=addbuf(buf,len*2)
  }
  buf[len]=a
  p+=2//uint16占2字节
  a=(*(*uint16)(unsafe.Pointer(p)))
 }
 str=string(utf16.Decode(buf[:len]))
 return
}

go和c类型装换

#cgo
char -->  C.char -->  byte
signed char -->  C.schar -->  int8
unsigned char -->  C.uchar -->  uint8
short int -->  C.short -->  int16
short unsigned int -->  C.ushort -->  uint16
int -->  C.int -->  int
unsigned int -->  C.uint -->  uint32
long int -->  C.long -->  int32 or int64
long unsigned int -->  C.ulong -->  uint32 or uint64
long long int -->  C.longlong -->  int64
long long unsigned int -->  C.ulonglong -->  uint64
float -->  C.float -->  float32
double -->  C.double -->  float64
wchar_t -->  C.wchar_t  -->  
void * -> unsafe.Pointer
#window
uint64 //对应SIZE_T
uint32 //对应DWORD
//整型数
//Go->C:
var i int
ci := C.int(i)
//C->Go:
var i C.int
goi := int(i)
//数组和字符串
//Go->C:
var str string
cstr := C.CString(str)
//C->Go:
/*
#include <stdlib.h>
#include <stdio.h>
char foo[] = "hellofoo";
char *bar = "hellobar";
*/
import "C"
import "fmt"
func main() {
    fmt.Printf("%s\n", C.GoString(&C.foo[0]))
    fmt.Printf("%s\n", C.GoString(C.bar))
}
//数组类型转换
/*
#include <stdio.h>
int cIArray[] = {1, 2, 3, 4, 5, 6, 7}; //int
float cFArray[] = {1.011, 2.022, 3.022, 4.023, 5.02, 6.03, 7.045};//float
*/
import "C"
import "fmt"
func main() {
    goIArray := C.cIArray[:]
    fmt.Println(goIArray)
    goFArray := C.cFArray[:]
    fmt.Println(goFArray)
}
//[]byte 转换为 *char
c_char := (*C.char)(unsafe.Pointer(&bt[0]))

fmt格式字符串:

打印格式 含义
%% 一个%字面量
%b 一个二进制整数值(基数为2),或者是一个(高级的)用科学计数法表示的指数为2的浮点数
%c 字符型。可以把输入的数字按照ASCII码相应转换为对应的字符
%d 一个十进制数值(基数为10)
%e 以科学记数法e表示的浮点数或者复数值
%E 以科学记数法E表示的浮点数或者复数值
%f 以标准记数法表示的浮点数或者复数值
%g 以%e或者%f表示的浮点数或者复数,任何一个都以最为紧凑的方式输出
%G 以%E或者%f表示的浮点数或者复数,任何一个都以最为紧凑的方式输出
%o 一个以八进制表示的数字(基数为8)
%p 以十六进制(基数为16)表示的一个值的地址,前缀为0x,字母使用小写的a-f表示
%q 使用Go语法以及必须时使用转义,以双引号括起来的字符串或者字节切片[]byte,或者是以单引号括起来的数字
%s 字符串。输出字符串中的字符直至字符串中的空字符(字符串以'\0‘结尾,这个'\0'即空字符)
%t 以true或者false输出的布尔值
%T 使用Go语法输出的值的类型
%U 一个用Unicode表示法表示的整型码点,默认值为4个数字字符
%v 使用默认格式输出的内置或者自定义类型的值,或者是使用其类型的String()方式输出的自定义值,如果该方法存在的话
%x 以十六进制表示的整型值(基数为十六),数字a-f使用小写表示
%X 以十六进制表示的整型值(基数为十六),数字A-F使用小写表示
  强制转换 隐式转换 类型断言
类型之间 可以 不可以 可以
类型->接口 可以 可以 可以
接口->类型 不可以 不可以 可以
接口之间 可以(编译期间确定) 可以(编译期间确定) 可以

补充:Golang常见类型转换

1.Type(expression):

int(time.Now().Weekday()) //星期转int
int(time.Now().Month())   //月份转int
var a float64
a = 3.1
b := int(a) //float64转int
var a int
a = 1
b := int64(a) //int转int64

2.strconv包:

string和int、int32、int64:

i, _ := strconv.Atoi(s) //string转int
s := strconv.Itoa(i)    //int转string
i, _ := strconv.ParseInt(s, 10, 32) //string转int32
ii := int32(I)
i, _ := strconv.ParseInt(s, 10, 64) //string转int32
s := strconv.FormatInt(i, 10)       //int64转string

ParseInt函数的官方介绍:

func ParseInt(s string, base int, bitSize int) (i int64, err error)

?返回字符串表示的整数值,接受正负号。

?base指定进制(2到36),如果base为0,则会从字符串前置判断,"0x"是16进制,"0"是8进制,否则是10进制。

?bitSize指定结果必须能无溢出赋值的整数类型,0、8、16、32、64 分别代表 int、int8、int16、int32、int64。

?返回的err是*NumErr类型的,如果语法有误,err.Error = ErrSyntax;如果结果超出类型范围err.Error = ErrRange。

FormatInt函数中的10,表示10进制。

string和float32、float64:

f, _ := strconv.ParseFloat(s, 32)        //string转float32
ff := float32(f)
f, _ := strconv.ParseFloat(s, 64)        //string转float64
s := strconv.FormatFloat(f, 'f', -1, 64) //float64转string

ParseFloat函数的官方介绍:

func ParseFloat(s string, bitSize int) (f float64, err error)

?解析一个表示浮点数的字符串并返回其值。

?如果s合乎语法规则,函数会返回最为接近s表示值的一个浮点数(使用IEEE754规范舍入)。

?bitSize指定了期望的接收类型,32是float32(返回值可以不改变精确值的赋值给float32),64是float64。

?返回值err是*NumErr类型的,语法有误的,err.Error=ErrSyntax;结果超出表示范围的,返回值f为±Inf,err.Error= ErrRange。

FormatFloat函数的官方介绍:

func FormatFloat(f float64, fmt byte, prec, bitSize int) string

?bitSize表示f的来源类型(32:float32、64:float64),会据此进行舍入。

?fmt表示格式:‘f'(-ddd.dddd)、‘b'(-ddddp±ddd,指数为二进制)、‘e'(-d.dddde±dd,十进制指数)、‘E'(-d.ddddE±dd,十进制指数)、‘g'(指数很大时用'e'格式,否则'f'格式)、‘G'(指数很大时用'E'格式,否则'f'格式)。

?prec控制精度(排除指数部分):对'f'、‘e'、‘E',它表示小数点后的数字个数;对'g'、‘G',它控制总的数字个数。如果prec 为-1,则代表使用最少数量的、但又必需的数字来表示f。

string和time:

t, _ := time.Parse("2006-01-02 15:04:05", s) //string转时间
t, _ := time.ParseInLocation("2006-01-02 15:04:05", s, time.Local) //string转本地时间
s := t.Format("2006-01-02 15:04:05")         //时间转string

3.类型断言:expression.(Type):

expression必须是接口类型,且自身类型与Type类型相符。

expression.(Type)的返回值一般为两个:value和ok,匹配成功ok为true,value有值,匹配失败ok为false,value无值;也可以直接接受value一个返回值,不过失败则直接panic:

func main() {
  var a interface{} = 100
  if aa, ok := a.(int); ok {
    fmt.Println(aa)
  }
}

还可以结合switch和case使用,来判断接口实际类型:

func main() {
  var t interface{} = 100
  switch i := t.(type) {
  case float32:
    fmt.Printf("i的类型%T i的值%v\n", i, i)
  case float64:
    fmt.Printf("i的类型%T i的值%v\n", i, i)
  case int:
    fmt.Printf("i的类型%T i的值%v\n", i, i)
  case bool:
    fmt.Printf("i的类型%T i的值%v\n", i, i)
  case string:
    fmt.Printf("i的类型%T i的值%v\n", i, i)
  default:
    fmt.Println("其他类型")
  }
}

4.JSON:

Golang中大多数数据类型都可以转化为有效的JSON文本,除了channel通道、complex复数、func函数等。

Golang指针可进行隐式转换,表面进行的是指针序列化,内部会针对指针进行取值操作,实际还是针对所指的对象进行序列化。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持三水点靠木。如有错误或未考虑完全的地方,望不吝赐教。

Golang 相关文章推荐
彻底理解golang中什么是nil
Apr 29 Golang
golang gopm get -g -v 无法获取第三方库的解决方案
May 05 Golang
解决Goland 同一个package中函数互相调用的问题
May 06 Golang
go语言基础 seek光标位置os包的使用
May 09 Golang
Golang实现AES对称加密的过程详解
May 20 Golang
Golang中异常处理机制详解
Jun 08 Golang
基于Go语言构建RESTful API服务
Jul 25 Golang
Golang中channel的原理解读(推荐)
Oct 16 Golang
Go语言安装并操作redis的go-redis库
Apr 14 Golang
Golang获取List列表元素的四种方式
Apr 20 Golang
GO中sync包自由控制并发示例详解
Aug 05 Golang
Go结合Gin导出Mysql数据到Excel表格
Aug 05 Golang
Golang: 内建容器的用法
May 05 #Golang
Go标准容器之Ring的使用说明
May 05 #Golang
go语言中GOPATH GOROOT的作用和设置方式
go设置多个GOPATH的方式
May 05 #Golang
浅谈golang 中time.After释放的问题
May 05 #Golang
golang 定时任务方面time.Sleep和time.Tick的优劣对比分析
May 05 #Golang
golang日志包logger的用法详解
May 05 #Golang
You might like
php数组函数序列之array_search()- 按元素值返回键名
2011/11/04 PHP
在win7中搭建Linux+PHP 开发环境
2014/10/08 PHP
php遍历类中包含的所有元素的方法
2015/05/12 PHP
ThinkPHP中获取指定日期后工作日的具体日期方法
2018/10/14 PHP
PHP中使用CURL发送get/post请求上传图片批处理功能
2018/10/15 PHP
北京奥运官方网站幻灯切换效果flash版打包下载
2008/01/30 Javascript
js定时调用方法成功后并停止调用示例
2014/04/08 Javascript
编写自己的jQuery提示框(Tip)插件
2015/02/05 Javascript
javascript 实现map集合
2015/04/03 Javascript
javascript中返回顶部按钮的实现
2015/05/05 Javascript
学习JavaScript设计模式之装饰者模式
2016/01/19 Javascript
基于jQuery的Web上传插件Uploadify使用示例
2016/05/19 Javascript
js实现弹窗居中的简单实例
2016/10/09 Javascript
微信公众平台开发教程(四) 实例入门:机器人回复(附源码)
2016/12/02 Javascript
深入理解JavaScript中的尾调用(Tail Call)
2017/02/07 Javascript
基于JQuery和原生JavaScript实现网页定位导航特效
2017/04/03 jQuery
基于JQuery的Ajax方法使用详解
2017/08/16 jQuery
浅入深出Vue之自动化路由
2019/08/06 Javascript
django将图片上传数据库后在前端显式的方法
2018/05/25 Python
opencv之为图像添加边界的方法示例
2019/12/26 Python
深入浅析Python 命令行模块 Click
2020/03/11 Python
python3 使用openpyxl将mysql数据写入xlsx的操作
2020/05/15 Python
Python生成器传参数及返回值原理解析
2020/07/22 Python
python 密码学示例——凯撒密码的实现
2020/09/21 Python
详解python定时简单爬取网页新闻存入数据库并发送邮件
2020/11/27 Python
5分钟弄清楚html5的drag and drop(小结)
2019/04/10 HTML / CSS
Ray-Ban雷朋奥地利官网:全球领先的太阳眼镜品牌
2020/10/12 全球购物
火山动力Java笔试题
2014/06/26 面试题
HttpServlet类中的主要方法都有哪些?各自的作用是什么?
2014/03/16 面试题
学校党的群众路线教育实践活动总结报告
2014/07/03 职场文书
2014离婚协议书范文
2014/09/10 职场文书
通知函的格式
2015/04/27 职场文书
企业开发CSS命名BEM代码规范实践
2022/02/12 HTML / CSS
Python自动化工具之实现Excel转Markdown表格
2022/04/08 Python
Java 中的 Lambda List 转 Map 的多种方法详解
2022/07/07 Java/Android
win10拖拽文件时崩溃怎么解决?win10文件不能拖拽问题解决方法
2022/08/14 数码科技