百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

unsafe—内存操作 un-buffered 内存

yuyutoo 2024-10-11 21:40 14 浏览 0 评论

unsafe包含绕过Go程序类型安全的操作。

导入unsafe的包可能不可移植,并且不受Go 1兼容性指南的保护。

unsafe.Pointer 四种特殊操作

Pointer表示指向任意类型的指针,Pointer类型有四种特殊操作不适用于其他类型:

  • 任何类型的指针值都可以转换为Pointer。
  • Pointer可以被转换为任何类型的指针值。
i:=unsafe.Pointer(new(interface{}))
*(*int)(i)=2
fmt.Println(*(*int)(i))
//输出:
3
  • uintptr可以转换为Pointer。
  • Pointer可以转换为uintptr。

因此Pointer可以无视类型系统,读写任意内存,应该特别小心使用。

unsafe.Pointer 六种使用模式

涉及Pointer的以下模式有效:(不使用这些模式的代码目前可能无效或将来无效)

即使是下面的有效模式也有重要的警告。

运行go vet可以帮助找到不符合这些模式的Pointer的使用,但是没有找到不代表代码一定有效

(1)将*T1指针转换为*T2。


如果T2不大于T1并且两者共享等效的内存布局,则此转换允许将一种类型的数据重新解释为另一种类型的数据。

例如math.Float64bits的实现:

func Float64bits(f float64) uint64 {
 return *(*uint64)(unsafe.Pointer(&f))
}

(2)将*Pointer 转换为*uintptr(不转回 *Pointer)


将Pointer转换为uintptr会产生指向内存地址的整数值,这种uintptr的通途通常是用来打印。

通常情况下将uintptr转换回Pointer是无效的,因为uintptr是一个整数,不是引用类型。

将Pointer转换为uintptr会创建一个没有指针语义的整数值。

即使uintptr保存某个对象的地址,如果对象移动,垃圾收集器也不会更新该uintptr的值,uintptr也不会保持该对象不被回收。

其余模式枚举了从uintptr到Pointer的唯一有效转换。

(3)使用算术将Pointer转换为uintptr并转回Pointer


如果p指向已分配的对象,则可以通过转换为uintptr,添加偏移量以及转换回指针来推进对象,如:

p = unsafe.Pointer(uintptr(p) + offset)

此模式最常见的用途是访问结构中的字段或数组的元素:

//等效于 f := unsafe.Pointer(&s.f)
f := unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + unsafe.Offsetof(s.f))
//等效于 e := unsafe.Pointer(&x[i])
e := unsafe.Pointer(uintptr(unsafe.Pointer(&x[0])) + i*unsafe.Sizeof(x[0]))

以这种方式添加和减去指针的偏移量都是有效的。

在所有情况下,结果必须继续指向原始分配的对象,否则:

var a = "hello"
b := &a
fmt.Println(*(*string)(unsafe.Pointer(^^uintptr(unsafe.Pointer(b)))),":",unsafe.Pointer(^uintptr(unsafe.Pointer(b))))
fmt.Println(*(*string)(unsafe.Pointer(^uintptr(unsafe.Pointer(b)))))
`输出:`
hello : 0xffffff3ffffefe1f
unexpected fault address 0xffffff3ffffefe1f
fatal error: fault
[signal SIGSEGV: segmentation violation code=0x1 addr=0xffffff3ffffefe1f pc=0x1091f4b]
// 可以看出第一条可以正常打印,第二条打印语句失败。

与C不同,将指针推进到原始分配的末尾是无效的,如:

var s typeName
end = unsafe.Pointer(uintptr(unsafe.Pointer(&s)) + unsafe.Sizeof(s))
fmt.Println("end:",*(*string)(end))
`输出:`
panic: runtime error: invalid memory address or nil pointer dereference

在分配空间之外的端点是无效的,如:

n := 2
b := make([]byte, n)
fmt.Println(len(b))
end := unsafe.Pointer(uintptr(unsafe.Pointer(&b[0])) + uintptr(n))
fmt.Println("end:", (*reflect.SliceHeader)(end).Len)
`输出:`
2
end: 734649689214812160

注意?: unsafe.Pointer和uintptr相互转换必须出现在一个表达式中,它们之间只有介入算术,在转换回Pointer之前,uintptr不能存储在变量中,如:

u := uintptr(p)
p = unsafe.Pointer(u + offset)

请注意,指针必须指向已分配的对象,因此它不会是nil,转换nil指针无效,如:

u := unsafe.Pointer(nil)
p := unsafe.Pointer(uintptr(u) + offset)

(4)在调用syscall.Syscall时将Pointer转换为uintptr


syscall中的Syscall函数将它们的uintptr参数直接传递给操作系统,然后操作系统可能会根据调用的详细信息将其中的一些重新解释为指针。

也就是说,系统调用实现隐式地将某些参数从uintptr转换回指针。

如果必须将指针参数转换为uintptr以用作参数,则该转换必须出现在调用表达式本身中:

syscall.Syscall(SYS_READ, uintptr(fd), uintptr(unsafe.Pointer(p)), uintptr(n))
//实现在:go1.12beta2/src/syscall/asm_linux_amd64.s

编译器将被引用的已经分配的对象保留,并且在调用完成之前不会移动,来处理在汇编中实现的函数调用的参数列表中将Pointer转换为uintptr的参数,即使仅从对象的类型中看起来在调用期间不再需要该对象。

要使编译器识别此模式,转换必须出现在参数列表中:

例如:

u := uintptr(unsafe.Pointer(p))
syscall.Syscall(SYS_READ, uintptr(fd), u, uintptr(n))
//在系统调用期间uintptr隐式转换回Pointer之前,uintptr不能存储在变量中

(5)将reflect.Value.Pointer或reflect.Value.UnsafeAddr的结果从uintptr转换为Pointer


reflect的名为Pointer和UnsafeAddr的方法返回类型是uintptr而不是unsafe.Pointer,是为了防止调用者在没有先导入unsafe包的情况下将结果更改为任意类型。

但是,这意味着结果很脆弱,并且必须像下面所示在调用后在同一表达式中立即转换为Pointer:

p := (*int)(unsafe.Pointer(reflect.ValueOf(new(int)).Pointer()))

与上述情况一样,在转换之前存储结果无效:

u := reflect.ValueOf(new(int)).Pointer()
p := (*int)(unsafe.Pointer(u))

(6)reflect.SliceHeader或reflect.StringHeader Data字段与Pointer相互转换


与(5)一样,反射数据结构SliceHeader和StringHeader将字段Data声明为uintptr,防止调用者在没有先导入unsafe包的情况下将结果更改为任意类型。

但是,这意味着SliceHeader和StringHeader仅在解释实际切片或字符串值的内容时有效。

var s = "hello"
var a = "你好"
hdr := (*reflect.StringHeader)(unsafe.Pointer(&s))
hdr.Data = (*reflect.StringHeader)(unsafe.Pointer(&a)).Data
hdr.Len = len(a) //长度必须不小于a,否则会截断
fmt.Printf("s:%v,len(s):%v\na:%v,len(a):%v\t", s, len(s), a, len(a))
`输出:`
s:你好,len(s):6
a:你好,len(a):6
//该种方法可以直接绕过类型系统,直接修改底层 Data 地址,
//避免在长字符串赋值时大量的内存拷贝,毕竟 Data 才8个字节。

上面例子中,hdr.Data实际上是一种引用字符串头中的底层指针uintprt。

通常,reflect.SliceHeader和reflect.StringHeader只能用作*reflect.SliceHeader和*reflect.StringHeader指向实际的切片或字符串,而不是普通的结构,程序不应声明或分配这些结构类型的变量。

例如:

var a = "你好"
var hdr reflect.StringHeader
hdr.Data = (*reflect.StringHeader)(unsafe.Pointer(&a)).Data
hdr.Len = len(a)
s:= *(*string)(unsafe.Pointer(&hdr)) //&a可能已经丢失

tips:类型声明和类型别名区别

package main
import "fmt"
type A = int
type I int
func main() {
 v := 100
 var a A = v // 不报错
 var i I = v // 编译报错,`cannot use v (type int) as type I in assignment`
 //但是可以强转,temp:=(I)(v) 
 //或 I--->int,
 //var v I
 //temp:=(int)(v)
 fmt.Println(a,i)
}

相关推荐

《保卫萝卜2》安卓版大更新 壕礼助阵世界杯

《保卫萝卜2:极地冒险》本周不仅迎来了安卓版本的重大更新,同时将于7月4日本周五,带来“保卫萝卜2”安卓版本世界杯主题活动的火热开启,游戏更新与活动两不误。一定有玩家会问,激萌塔防到底进行了哪些更新?...

儿童手工折纸:胡萝卜,和孩子一起边玩边学carrot

1、准备两张正方形纸,一橙一绿,对折出折痕。2、橙色沿其中一条对角线如图折两三角形。3、把上面三角折平,如图。4、绿色纸折成三角形。5、再折成更小的三角形。6、再折三分之一如图。7、打开折纸,压平中间...

《饥荒》食物代码有哪些(饥荒最新版代码总汇食物篇)

饥荒游戏中,玩家们需要获取各种素材与食物,进行生存。玩家们在游戏中,进入游戏后按“~”键调出控制台使用代码,可以直接获得素材。比如胡萝卜的代码是carrot,玉米的代码是corn,南瓜的代码是pump...

Skyscanner:帮你找到最便宜机票 订票不求人

你喜欢旅行吗?在合适的时间、合适的目的地,来一场说走就走的旅行?机票就是关键!Skyscanner这款免费的手机应用,在几秒钟内比较全球600多家航空公司的航班安排、价格和时刻表,帮你节省金钱和时间。...

小猪佩奇第二季50(小猪佩奇第二季英文版免费观看)

Sleepover过夜Itisnighttime.现在是晚上。...

我在民政局工作的那些事儿(二)(我在民政局上班)

时间到了1997年的秋天,经过一年多的学习和实践,我在处理结婚和离婚的事情更加的娴熟,也获得了领导的器重,所以我在处理平时的工作时也能得心应手。这一天我正在离婚处和同事闲聊,因为离婚处几天也遇不到人,...

夏天来了就你还没瘦?教你不节食13天瘦10斤的哥本哈根减肥法……

好看的人都关注江苏气象啦夏天很快就要来了你是否和苏苏一样身上的肉肉还没做好准备?真是一个悲伤的故事……下面这个哥本哈根减肥法苏苏的同事亲测有效不节食不运动不反弹大家快来一起试试看吧~DAY1...

Pursuing global modernization for peaceful development, mutually beneficial cooperation, prosperity for all

AlocalworkeroperatesequipmentintheChina-EgyptTEDASuezEconomicandTradeCooperationZonei...

Centuries-old tea road regains glory as Belt and Road cooperation deepens

FUZHOU/ST.PETERSBURG,Oct.2(Xinhua)--NestledinthepicturesqueWuyiMountainsinsoutheastChi...

15 THE NUTCRACKERS OF NUTCRACKER LODGE (CONTINUED)胡桃夹子小屋里的胡桃夹子(续篇)

...

AI模型部署:Triton Inference Server模型部署框架简介和快速实践

关键词:...

Ftrace function graph简介(flat function)

引言由于android开发的需要与systrace的普及,现在大家在进行性能与功耗分析时候,经常会用到systrace跟pefetto.而systrace就是基于内核的eventtracing来实...

JAVA历史版本(java各版本)

JAVA发展1.1996年1月23日JDK1.0Java虚拟机SunClassicVM,Applet,AWT2.1997年2月19日JDK1.1JAR文件格式,JDBC,JavaBea...

java 进化史1(java的进阶之路)

java从1996年1月第一个版本诞生,到2022年3月最新的java18,已经经历了27年,整整18个大的版本。很久之前有人就说java要被淘汰,但是java活到现在依然坚挺,不知道java还能活...

学习java第二天(java学完后能做什么)

#java知识#...

取消回复欢迎 发表评论: