Rust中的特殊类型所占的内存大小
创始人
2024-11-14 06:39:12

可以使用std::mem:size_of获取类型大小:

use std::mem::size_of;  struct Journal(String, u32); trait Summary {} impl Summary for Journal {}  fn main() {     println!("普通结构体相关:");     println!("{}", size_of::<&Journal>());     println!("{}", size_of::>());      println!("特征对象相关:");     println!("{}", size_of::>());     println!("{}", size_of::<&dyn Summary>());     println!("{}", size_of::<&mut dyn Summary>());     println!("{}", size_of::<*const dyn Summary>());     println!("{}", size_of::<*mut dyn Summary>());      println!("切片相关:");     println!("{}", size_of::<&str>());     println!("{}", size_of::<&[i32]>());     println!("{}", size_of::<&[i32; 100]>());      println!("集合相关:");     println!("{}", size_of::<[i32; 100]>());     println!("{}", size_of::>());     println!("{}", size_of::()); }  

输出

普通结构体相关: 8 8 特征对象相关: 16 16 16 16 16 切片相关: 16 16 8 集合相关: 400 24 24  

原因分析

Rust中,一个普通指针的大小为usize,与计算机CPU字长相等,对64位计算机来说usize=64/8=8字节

  • 对普通结构体来说,其引用/Box智能指针所占大小为usize,指示在内存中的起始位置即可,因为普通结构体编译时大小确定,运行时存放在栈上
  • 特征对象非常特殊,其引用/Box智能指针/裸指针占的大小均为2*usize
    • 这是因为特征对象编译时大小不确定,因此需要在运行时动态获知对象实例以及特征方法在内存中的位置,因此需要两个指针,分别指向它们
  • 对切片类型:
    • &str所占空间为2*usize,所引用字符串可能来自于堆内存/全局数据区,&str指示了所引用数据的起始位置和长度
    • &[i32]&str类似(&str是对&[u8]的封装),所占空间为2*usize,指示了所引用字符串在堆内存/栈内存中的起始位置和长度
    • &[i32; N]仅占一个usize,因为[i32; N]编译时大小已知,存放在栈上,&[i32; N]不需要像&[i32]那样额外记录长度
  • 对集合相关:
    • [i32; 100]编译时大小已知,存放在栈上,内存大小为4*100=400个字节
    • Vec动态数组的数据存放在堆上,Vec类型占三个usize的长度,分别指示数组在堆中的起始位置,数组长度以及数组当前的最大容量
    • String是对Vec的封装,其数据存放在堆上,String类型占三个usize的长度,分别指示字符串在堆中的起始位置,字符串长度以及字符串当前的最大长度

其实,VecString的本质都是智能指针,它们的大小都是固定的(大小不固定也没法通过编译啊),运行时存放在栈上。但是它们所包裹数据的长度是可变的,存放在堆上。它们的本质是通过存放在栈上的指针访问堆内存,来操作堆上包裹的数据。

综上所述,智能指针/引用所占内存大小是固定的,但是其指向的数据大小未必固定。对于所占内存固定已知的数据类型,运行时会把它们放在栈上,对于所占内存不固定的数据类型,运行时会把它们放在堆上,然后可以通过存放在栈上的指针来操作堆上的数据。

相关内容

热门资讯

裸辞做“一人公司”,我后悔了 去年这个时候,一位以色列程序员正在东南亚旅行。他顺手把一个在脑子里转了很久的想法做成了产品,一个让任...
南京建成国内首个Pre-6G试... 4月21日,2026全球6G技术与产业生态大会在南京开幕。全息互动技术展台前,一名远在北京的工作人员...
超梵求职受邀参加“2025抖音... 超梵求职受邀参加“2025抖音巨量引擎成人教育行业生态大会”,探讨分享优质内容传播,服务万千学员。 ...
摩托罗拉Razr 2026(R... IT之家 4 月 22 日消息,摩托罗拉宣布新一代 Razr 折叠手机将于 4 月 29 日在美国发...
库克卸任,特纳斯领航:苹果新纪... 苹果首席执行官蒂姆·库克将卸任,硬件工程主管约翰·特纳斯将接任,苹果公司今天宣布此事。 库克将在夏季...