欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 教育 > 幼教 > 【Rust中级教程】1.5. 内存 Pt.3:深入探究Rust堆内存底层实现

【Rust中级教程】1.5. 内存 Pt.3:深入探究Rust堆内存底层实现

2025/9/23 23:43:31 来源:https://blog.csdn.net/weixin_71793197/article/details/145583400  浏览:    关键词:【Rust中级教程】1.5. 内存 Pt.3:深入探究Rust堆内存底层实现

喜欢的话别忘了点赞、收藏加关注哦(加关注即可阅读全文),对接下来的教程有兴趣的可以关注专栏。谢谢喵!(=・ω・=)
请添加图片描述

1.5.1. 堆内存(Heap)

  • Heap意味着混乱,而stack则相对比较整齐。
  • Heap是一个内存池,并没有绑定到当前程序的调用栈,而stack绑定到当前程序的调用栈。
  • Heap是为在编译时没有已知大小的类型准备的,而stack上的数据在编译时大小必须已知。
    请添加图片描述

如图,heap上的数据存放的位置和自身的大小都是不一定的,比较常见的情况是stack有一个指针指向heap上的数据。

什么叫在编译时大小不已知?

  • 一些类型会随着时间变大或变小,例如StringVec<T>
  • 另一些类型的大小不会改变,但是无法告诉编译器需要分配多少内存
  • 另一个例子是trait对象(详见 【Rust自学】19.5.4. 动态大小和和Sized trait),它允许程序员模拟一些动态语言的特性——将多个类型放进一个容器
  • 以上三种都叫动态大小类型(dynamically sized types,简称DST)

Heap允许你显示地分配连续的内存块。当这么做时,你就会得到一个指针,它指向内存开始的地方。

Heap中的值会一直有效,直到你对它显式地释放。如果你想让值活得比当前函数frame(详见 1.4. 内存 Pt.2)的生命周期还长,就很有用。如果值是函数的返回值,那么调用函数可以在它的stack上留一些空间给被调用函数让它把值在返回前写入进去。

1.5.2. 堆内存与线程安全

如果想要把值送到另一个线程,当前线程可能根本无法与那个线程共享stack frames,你就可以把它存在堆内存上。因为函数返回时堆内存上的分配不会消失,所以你在一个地方为值分配内存,把指向它的指针传给另一个线程,就可以让那个线程安全地操作于这个值。

换一种说法:当你分配堆内存时,结果指针会有一个无约束的生命周期,你的程序想让数据活多久都行。

1.5.3. 堆内存的交互机制

堆内存上的变量必须通过指针访问。看个例子:

fn main(){  let a: i32 = 40; // Stack  let b: Box<i32> = Box::new(60); // Heap  let result = a + b;  let result = a + *b;  println!("{} + {} = {}", a, b, result);  
}
  • ai32类型,放在栈内存上的
  • bBox<i32>类型,放在堆内存上的

但是这么写肯定是有问题的,问题出在let result = a + b;上:堆内存上的数据必须通过指针来访问,b是指针,a是数字,两者类型不同肯定不能相加。

所以我们删掉这句话把原代码改成:

fn main(){  let a: i32 = 40; // Stack  let b: Box<i32> = Box::new(60); // Heap  let result = a + *b;  println!("{} + {} = {}", a, b, result);  
}

let result = a + *b;通过*b进行了解引用,把指针指向的值60取出来了。

输出:

40 + 60 = 100

Rust与堆内存的交互方式

Rust里与堆内存交互的主要方式就是Box<T>类型。

当我们使用Box::new函数创建类型为Box<T>的实例时,值(传进Box::new函数的参数)就会被放在堆内存上,返回的(BOx<T>)就是指向堆内存上的指针。当Box被丢弃时,内存就会被释放。

如果忘记释放堆内存,就会导致内存泄漏。但有时候程序员会故意让内存泄漏,例如有一个只读的配置,整个程序都需要访问它。这时候就可以通过Box::leak得到一个‘static引用,从而显式地让其进行泄露。

看个例子:

use std::mem::drop;  fn main(){  let a = Box::new(1);  let b = Box::new(1);  let c = Box::new(1);  let result1 = *a + *b + *c;  drop(a);  let d = Box::new(1);  let result2 = *b + *c + *d;  println!("{} {}", result1, result2);  
}
  • 使用std::mem::drop函数可以手动释放

我们讲讲这个程序的逻辑:

  • 首先声明了变量abc,值都是存储在堆内存上的1(Box<i32>类型)
  • 使用*对三个变量都进行解引用然后相加得到result1
  • 得到result1之后使用drop函数直接抛弃了a
  • 然后声明了变量d,值是存储在堆内存上的1(Box<i32>类型)
  • 使用对*三个变量(bcd)进行解引用然后相加得到result2
  • 打印出result1result2

我们用一张图来看看这个程序执行中内存的变化:
请添加图片描述

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词