gpt4 book ai didi

rust - Rust 移动语义实际上是如何工作的

转载 作者:行者123 更新时间:2023-12-05 03:37:25 28 4
gpt4 key购买 nike

在我写了一些代码和阅读了一些文章之后,我对 Rust 中的移动语义有点困惑,我认为值移动后,它应该被释放,内存应该是无效的。所以我尝试写一些代码来作证。

第一个例子

#[derive(Debug)]
struct Hello {
field: u64,
field_ptr: *const u64,
}

impl Hello {
fn new() -> Self {
let h = Hello {
field: 100,
field_ptr: std::ptr::null(),
};
h
}

fn init(&mut self) {
self.field_ptr = &self.field as *const u64;
}
}
fn main(){
let mut h = Hello::new();
h.init();
println!("=================");
println!("addr of h: ({:?}) \naddr of field ({:?})\nfield_ptr: ({:?}) \nptr value {:?}", &h as *const Hello, &h.field as *const u64, h.field_ptr, unsafe {*h.field_ptr});

let c = &h.field as *const u64;
let e = &h as *const Hello;
let a = h;
let d = &a.field as *const u64;

println!("=================");
println!("addr of a: ({:?}) \naddr of field ({:?})\nfield_ptr: ({:?}) \nptr value {:?}", &a as *const Hello, &a.field as *const u64, a.field_ptr, unsafe {*a.field_ptr});
println!("=================");
println!("addr of c {:?}\nvalue {:?}", c, unsafe {*c});
println!("addr of d {:?}\nvalue {:?}", d, unsafe {*d});
println!("addr of e {:?}\nvalue {:?}", e, unsafe {&*e});
}

上面代码的结果是

=================
addr of h: (0x7ffee9700628)
addr of field (0x7ffee9700628)
field_ptr: (0x7ffee9700628)
ptr value 100
=================
addr of a: (0x7ffee9700720)
addr of field (0x7ffee9700720)
field_ptr: (0x7ffee9700628)
ptr value 100
=================
addr of c 0x7ffee9700628
value 100
addr of d 0x7ffee9700720
value 100
addr of e 0x7ffee9700628
value Hello { field: 100, field_ptr: 0x7ffee9700628 }

所以,我创建了一个自引用结构体 Hello 并使 field_ptr 指向 u64 字段,并使用原始指针来保存结构体的地址和字段的地址,然后移动 h到 a 使 h 变量无效,但我仍然可以通过原始点获取 IMO 不应该存在的原始变量的值?

第二个例子

struct Boxed {
field: u64,
}
fn main(){

let mut f = std::ptr::null();
{
let boxed = Box::new(Boxed{field: 123});
f = &boxed.field as *const u64;
}
println!("addr of f {:?}\nvalue {:?}", f, unsafe {&*f});
}

结果

addr of f 0x7fc1f8c05d30
value 123

我创建了一个装箱值并在使用原始点保存它的地址后将其删除,我仍然可以通过原始点读取它的字段值。

所以我的困惑是

  1. rust 中的移动实际上是一个 memcpy 吗?而原始变量只是被编译器“隐藏”了?
  2. rust 什么时候真正释放变量在堆上的内存? (第二个例子)

谢谢

我读过的 How does Rust provide move semantics?

最佳答案

所以你输出的第一 block 应该很清楚,对吧?结构的地址只是该结构在内存中所在位置的第一位内存,这与其第一个字段的地址相同。

现在开始你的第二个街区。您正在获取结构中的一些原始指针,然后通过 let a = h 移动结构。

它所做的是:在堆栈上,我们现在有一个新变量a,它是变量h 是。这就是 aa.field 都有新地址的原因。当然,原始指针仍然指向旧的 h.field 地址,这就是您仍然可以访问该数据的原因。

请注意,您只能通过 unsafe block 来执行此操作,因为您所做的 是不安全的。无法保证您的字段指针指向的任何内容都将保持有效。

如果您删除所有对不安全 结构的使用,将无法通过h.field 访问a.field

同样的想法适用于第二个例子。如果您不使用原始指针和不安全的 block ,您将无法获得丢弃的东西,那是因为这段代码非常可疑。在您的简单示例中,它仍然有效,因为 Rust 不会继续并扰乱已删除值的内存。除非您的程序中有其他东西重新利用该内存,否则它将保持您离开时的状态。

关于rust - Rust 移动语义实际上是如何工作的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69336466/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com