Golang 中的 unsafe 包详解

Golang 中的 unsafe 包用于在运行时进行低级别的操作。这些操作通常是不安全的,因为可以打破 Golang 的类型安全性和内存安全性,使用 unsafe 包的程序可能会影响可移植性和兼容性。接下来看下 unsafe 包中的类型和函数。

unsafe.Pointer 类型

通常用于类型转换和指针运算,定义如下:

type Pointer *ArbitraryType

可以将其他类型的指针转换为unsafe.Pointer类型,以进行低级别的操作。看个简单示例:

package main
 
import (
	"fmt"
	"unsafe"
)
 
func main() {
	i := 30
	ptr1 := &i
 
	var ptr2 *int64 = (*int64)(unsafe.Pointer(ptr1))
	*ptr2 = 8
 
	fmt.Println(i)
}

unsafe.ArbitraryType 类型

定义如下:

type ArbitraryType int

ArbitraryType 仅用于文档目的,实际上并不是 unsafe 包的一部分,用于表示任意 Go 表达式的类型。

unsafe.IntegerType 类型

定义如下:

type IntegerType int

IntegerType 仅用于文档目的,实际上并不是 unsafe 包的一部分,用于表示任意整数类型。

unsafe.Add 函数

定义如下:

func Add(ptr Pointer, len IntegerType) Pointer

用于进行指针的加法运算,将一个指针和指定的偏移量相加,得到一个新的指针。简单示例如下:

package main

import (
	"fmt"
	"unsafe"
)

func main() {
	arr := []int{1, 2, 3, 4}
	index := 2
	ptr := unsafe.Pointer(&arr[0])
	newPtr := unsafe.Add(ptr, uintptr(index)*unsafe.Sizeof(arr[0]))
	cc := (*int)(newPtr)
	fmt.Println(*cc)
	fmt.Println(newPtr)
}

unsafe.Sizeof 函数

定义如下:

func Sizeof(x ArbitraryType) uintptr

用于获取一个类型或值的字节大小。简单示例如下:

package main

import (
	"fmt"
	"unsafe"
)

func main() {
	var arr [5]int
	fmt.Println(unsafe.Sizeof(arr)) // 输出: 40
	fmt.Println(unsafe.Sizeof(0))   // 输出: 8
}

unsafe.Slice 函数

定义如下:

func Slice(ptr *ArbitraryType, len IntegerType) []ArbitraryType

用于创建一个与原始数组共享底层数据的切片。简单示例如下:

package main

import (
	"fmt"
	"unsafe"
)

func main() {
	var arr [5]int
	arr[0] = 10
	arr[1] = 20
	arr[2] = 30
	arr[3] = 40
	arr[4] = 50

	slice := unsafe.Slice(&arr[0], 3)
	fmt.Println(slice) // 输出: [10 20 30]
}

定义了一个包含5个整数的数组 arr,然后使用 unsafe.Slice 函数创建了一个从索引0开始、长度为3的切片 slice。

unsafe.Offsetof

定义如下:

func Offsetof(x ArbitraryType) uintptr

作用是返回结构体成员在内存中的位置离结构体起始处(结构体的第一个字段的偏移量都是0)的字节数。简单示例如下:

package main

import (
	"fmt"
	"unsafe"
)

type MyStruct struct {
	Field1 int64
	Field2 string
}

func main() {
	var myStruct MyStruct
	fmt.Println(unsafe.Offsetof(myStruct.Field1)) // 输出: 0
	fmt.Println(unsafe.Offsetof(myStruct.Field2)) // 输出: 8
}

unsafe.Alignof

定义如下:

func Alignof(x ArbitraryType) uintptr

返回参数对齐所需的字节数。对于某些结构体,其字段的内存对齐可能会影响到结构体所占用的内存大小。简单示例如下:

package main

import (
	"fmt"
	"unsafe"
)

type MyStruct struct {
	Field1 int
	Field2 string
}

func main() {
	fmt.Println(unsafe.Alignof(int(0)))     // 输出: 8
	fmt.Println(unsafe.Alignof(string(""))) // 输出: 8
	fmt.Println(unsafe.Alignof(MyStruct{})) // 输出: 8
}

小结

unsafe 包提供了一些低级别的操作,使用的时候一定要谨慎,必须非常了解自己在做什么,以免引入不安全的问题。

相关推荐
硬件人某某某4 分钟前
基于Django的手办交易平台~源码
后端·python·django
Mr.Wang8096 分钟前
条款23:宁以non-member、non-friend替换member函数
开发语言·c++
关关钧22 分钟前
【R语言】读取CSV数据时,显示[1] PK...<0 行> (或0-长度的row.names)
开发语言·r语言
以卿a1 小时前
C++ 模板初阶
开发语言·c++
s:1031 小时前
【框架】参考 Spring Security 安全框架设计出,轻量化高可扩展的身份认证与授权架构
java·开发语言
道不尽世间的沧桑2 小时前
第17篇:网络请求与Axios集成
开发语言·前端·javascript
久绊A2 小时前
Python 基本语法的详细解释
开发语言·windows·python
软件黑马王子6 小时前
C#初级教程(4)——流程控制:从基础到实践
开发语言·c#
闲猫6 小时前
go orm GORM
开发语言·后端·golang
丁卯4046 小时前
Go语言中使用viper绑定结构体和yaml文件信息时,标签的使用
服务器·后端·golang