登录
首页 >  Golang >  Go问答

在 golang 中写入但不读取内存位置时出现 SIGSEGV

来源:stackoverflow

时间:2024-04-07 23:00:33 162浏览 收藏

最近发现不少小伙伴都对Golang很感兴趣,所以今天继续给大家介绍Golang相关的知识,本文《在 golang 中写入但不读取内存位置时出现 SIGSEGV》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

问题内容

我的印象是使用 unsafe 包允许您读取/写入任意数据。我正在尝试更改 interface{} 指向的值而不更改指针本身。

假设 interface{} 实现为

type _interface struct {
    type_info *typ
    value unsafe.pointer
}

尽管读取成功,但设置失败并出现 sigsegv。

func data(i interface{}) unsafe.Pointer {
    return unsafe.Pointer((*((*[2]uintptr)(unsafe.Pointer(&i))))[1])
}

func main() {
    var i interface{}
    i = 2
    fmt.Printf("%v, %v\n", (*int)(data(i)), *(*int)(data(i)))
    *((*int)(data(i))) = 3
}

我做错了什么,还是这在golang中不可能?


解决方案


嗯......这是我目前如何理解你的第二个代码示例,以防万一我犯了错误(如果你发现我所描述的内容有任何问题,我的答案可能是不可挽回的错误,你应该忽略其余的我要说的)。

  1. main中的接口i分配内存。
  2. i 的值设置为整数类型,值为 2。
  3. data中的接口i分配内存。
  4. 将main的i的值复制到data的i;即设置新接口的值为整数类型,值为2。
  5. 将新变量的地址转换为指向 uintptr 的长度为 2 的数组的指针(以 unsafe.Pointer 充当强制编译器接受此转换的中介)。
  6. 将数组的第二个元素(其值是 datai 的值部分的地址)转换回 unsafe.Pointer 并返回它。

我已经做了 an attempt 在更多步骤中做同样的事情,但不幸的是我遇到了所有相同的问题:程序识别出我有一个非零指针,并且它能够取消引用该指针以进行读取,但使用相同的指针写入会产生运行时错误。

go vet 抱怨的是第 6 步,我认为这是因为,根据 the package docs,

更重要的是,据我所知(尽管我承认我在不扫描编译器和运行时源代码的情况下挖掘显式确认时遇到了麻烦),运行时似乎并没有跟踪interface{} 类型作为离散指针,具有自己的引用计数;当然,您可以通过 writing another interface valueinterface{} 的单词践踏成整个事情,但这似乎根本不是您想要做的(写入位于界面类型,全部无需移动指针)。

有趣的是,我们似乎可以通过定义我们自己的结构化类型来实现,而编译器并没有对它进行特殊处理(接口显然有些特殊,具有类型断言语法)和所有)。也就是说,我们可以使用 unsafe.Pointer 来维护一个指向内存中特定点的引用,并且无论我们将其转换为什么,即使值发生变化,内存地址也永远不会移动(并且可以通过转换来重新解释该值)它到其他东西)。让我有点惊讶的部分是,至少在我自己的示例中,并且至少在演示环境中,所指向的值似乎没有固定的大小;我们可以建立一个地址来写入一次,即使数据量很大(或很小),对该地址的重复写入也会成功。

当然,至少在这个实现中,我们失去了许多与接口类型相关的其他好东西,尤其是非空接口类型(即与方法)。因此,没有办法使用它来(例如)创建一个超级偷偷摸摸的“通用”类型。看起来接口就是它自己的值,并且该值的定义的一部分是内存中的地址,但它与指针并不完全相同。

本篇关于《在 golang 中写入但不读取内存位置时出现 SIGSEGV》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>