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

unsafe—内存操作 un-buffered 内存

yuyutoo 2024-10-11 21:40 3 浏览 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)
}

相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: