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

相关推荐
捕鲸叉11 分钟前
创建线程时传递参数给线程
开发语言·c++·算法
A charmer15 分钟前
【C++】vector 类深度解析:探索动态数组的奥秘
开发语言·c++·算法
码农小旋风16 分钟前
详解K8S--声明式API
后端
Peter_chq18 分钟前
【操作系统】基于环形队列的生产消费模型
linux·c语言·开发语言·c++·后端
Yaml441 分钟前
Spring Boot 与 Vue 共筑二手书籍交易卓越平台
java·spring boot·后端·mysql·spring·vue·二手书籍
小小小妮子~42 分钟前
Spring Boot详解:从入门到精通
java·spring boot·后端
hong16168844 分钟前
Spring Boot中实现多数据源连接和切换的方案
java·spring boot·后端
记录成长java2 小时前
ServletContext,Cookie,HttpSession的使用
java·开发语言·servlet
前端青山2 小时前
Node.js-增强 API 安全性和性能优化
开发语言·前端·javascript·性能优化·前端框架·node.js
睡觉谁叫~~~2 小时前
一文解秘Rust如何与Java互操作
java·开发语言·后端·rust