从字符串使用看Golang和Rust对内存使用的区别

devtools/2025/1/14 13:54:28/

从字符串使用看Golang和Rust对内存使用的区别

​ 今天从Rust偶然回到Golang的世界,怎么写代码怎么别扭,总是忍不住在句子结尾加个分号…看到golang的字符串使用起来特别爽可以到处复制疯狂乱用,有一种从部队宿舍豆腐块被子的生活回归到居家肥宅的随意感,想起好久之前看的golang底层有关的内容,就写点东西来比较一下golangrust对string的使用。

Go的字符串

在 Go 中,每个字符串本质上是一个结构体,其定义好了就不可用索引进行修改,包含两个字段:

  1. 指向字符串内容的指针*byte):8 字节(在64 位架构下)。
  2. 字符串的长度len):8 字节。

其内存布局是这个这样的结构体:

golang">struct string {data uintptr // 指向字符串内容的指针len  int     // 字符串的长度
}

所以golang里面的一个string事实上占用的正真大小为:

golang">package mainimport ("fmt""unsafe"
)func main() {str := "hello"fmt.Printf("len(): %d bytes (content size)\n", len(str))fmt.Printf("Sizeof string struct: %d bytes\n", unsafe.Sizeof(str))fmt.Printf("Total estimated memory: %d bytes\n", unsafe.Sizeof(str)+uintptr(len(str)))
}

output:

String: hello
len(): 5 bytes (content size)
Sizeof string struct: 16 bytes
Total estimated memory: 21 bytes

Rust的字符串

String

rust里面有两种常用的字符串,一个是String,另一个是&str。

在Rust中,String是一个可变的、堆分配的类型,底层实现是一个Vec<u8>

rust">pub struct String {vec: Vec<u8>,
}

所以一个String本质上还包含着vector的结构,也就是:

  1. 指向堆分配数据的指针:8 字节(在64 位系统上)。
  2. 字符串的长度usize):8 字节。
  3. 堆分配容量usize):8 字节。

所以说一个rust的string所占用的内存就至少是24字节,而且其本质由于就是一个vector,可以根据索引修改vector里面的值

rust">fn main() {let s = String::from("hello");println!("Size of String struct: {} bytes", std::mem::size_of::<String>());println!("Content length: {} bytes", s.len());
}

output:

Size of String struct: 24 bytes
Content length: 5 bytes

&str

​ 另一个是&str,在Rust中,&str是一个字符串切片类型,它是对字符串数据的不可变引用。相比于String&str更轻量级,因为它只是一个指向实际字符串数据的引用,而不是负责管理字符串数据本身。简单来说&str就是一个对静态内存或者堆内存的一个引用。一个&str的大小是固定的,包含两个部分:

  1. 指向字符串内容的指针*const u8):8 字节(在 64 位系统上)。
  2. 字符串的长度usize):8 字节。

所以一个&str至少就是16字节。

rust">fn main() {let s = "hello"; // &str 类型println!("Size of &str: {} bytes", std::mem::size_of_val(&s));println!("Content length: {} bytes", s.len());
}

output:

Size of &str: 16 bytes
Content length: 5 bytes

why

&strString的关系

  1. String转换为&str

    • &str是对String数据的不可变引用。

    • 通过&操作可以将String转换为&str,这并不是简单的取地址,而是生成一个指向String内部数据的引用。

    • 示例:

      let s = String::from("hello");
      let slice: &str = &s; //将 String 转为&str
      println!("{}", slice);
      
  2. &str转换为String

    • 如果你需要一个拥有所有权的字符串,可以通过.to_string()String::from()&str转换为String

    • 示例:

      let slice: &str = "hello";
      let s: String = slice.to_string(); // 将 &str 转为 String
      println!("{}", s);
      

怎么要两个字符串?

都有String了,为什么还要个这种&str,有时候看别人的代码都只创建&str而不是String,这是为什么呢?而且String还可以修改可以直接克隆。

  1. 轻量级和高效
  • 内存开销更小

    • &str是不可变的引用,不需要额外的堆分配。
    • 它只包含一个指针和一个长度,总大小为16 字节,比String24 字节更小。
  • 数据共享

    • &str是对现有字符串数据的引用,不会创建新数据或重新分配内存。适用于只读场景,避免不必要的性能开销。
    • 例如,字符串字面量("hello")是静态分配的,用&str表示效率更高。
  • 性能优越

    • 在函数参数中使用&str而不是String,避免堆分配和拷贝。

    • 示例:

      rust">fn greet(name: &str) {println!("Hello, {}!", name);
      }
      let name = String::from("Alice");
      greet(&name); // 传递不可变引用,避免拷贝,类似于golang里面传递&string
      
  1. 安全性
  • &str的不可变性提供了额外的安全保障,确保引用的数据不会意外被修改。

3.适配静态字符串

  • 如果数据是静态的(如程序中的字符串字面量),选择&str是合适的,经常作为全局静态变量使用

    let s: &str = "hello world"; // 静态字符串
    

结尾

总结对比

特性RustString&strGolangstring
大小24字节16字节16字节
内存管理动态分配堆内存引用已有数据堆分配
是否可变可变不可变不可变
用途动态字符串管理,修改内容高效只读,数据共享很多
典型场景动态构建和管理字符串静态字符串,函数参数,全局变量很多

http://www.ppmy.cn/devtools/150422.html

相关文章

监控易钢铁行业:IT 设备、动环设施全方位一体化监控

在当今数字化浪潮汹涌澎湃的时代&#xff0c;各行业对信息技术的依赖程度日益加深&#xff0c;钢铁行业亦不例外。作为钢铁行业的中流砥柱&#xff0c;某钢铁公司在贸易、信息科技、循环经济和物流等多元业务领域开疆拓土&#xff0c;取得了斐然佳绩。 然而&#xff0c;随着业务…

【华为OD-E卷 - 篮球比赛 100分(python、java、c++、js、c)】

【华为OD-E卷 - 篮球比赛 100分&#xff08;python、java、c、js、c&#xff09;】 题目 篮球(5V5)比赛中&#xff0c;每个球员拥有一个战斗力&#xff0c;每个队伍的所有球员战斗力之和为该队伍的总体战斗力。 现有10个球员准备分为两队进行训练赛&#xff0c;教练希望2个队…

【PPTist】插入形状、插入图片、插入图表

一、插入形状 插入形状有两种情况&#xff0c;一种是插入固定的形状&#xff0c; 一种是插入自定义的形状。 插入固定的形状时&#xff0c;跟上一篇文章 绘制文本框 是一样一样的&#xff0c;都是调用的 mainStore.setCreatingElement() 方法&#xff0c;只不多传的类型不一…

PHP语言的学习路线

PHP语言的学习路线 PHP&#xff08;Hypertext Preprocessor&#xff09;是一种广泛使用的开源服务器端脚本语言&#xff0c;尤其适用于Web开发。由于其易学易用、功能强大&#xff0c;PHP成为了许多动态网站和Web应用程序开发的首选语言。随着Web3.0和云计算的兴起&#xff0c…

【数据仓库】— 5分钟浅谈数据仓库(适合新手)从理论到实践

大家好&#xff0c;我是摇光~ 对于刚进入大数据领域的萌新&#xff0c;且想要在数据分析岗、数据运维岗、数据工程师这些岗位立足&#xff0c;了解数据仓库是必要的&#xff0c;接下来我尽量用通俗易懂的语言让大家了解到数据仓库。 在当今大数据盛行的时代&#xff0c;数据仓…

支持selenium的chrome driver更新到131.0.6778.264

最近chrome释放新版本&#xff1a;131.0.6778.264 如果运行selenium自动化测试出现以下问题&#xff0c;是需要升级chromedriver才可以解决的。 selenium.common.exceptions.SessionNotCreatedException: Message: session not created: This version of ChromeDriver only s…

Ubuntu 20.04 安装Cuda 12.2版本踩坑记录

Ubuntu 20.04 安装Cuda 12.2版本踩坑记录 文章目录 Ubuntu 20.04 安装Cuda 12.2版本踩坑记录查看Ubuntu版本不成功的方式&#xff1a;使用deb安装卸载现有的 NVIDIA 驱动&#xff1a;安装符合要求的驱动版本&#xff1a; 成功的安装方式&#xff1a;使用runfile安装user账户nvc…

macOS 版本对应 Xcode 版本,以及 Xcode 历史版本下载

注&#xff1a;当前页面的所有Xcode下载链接均为苹果官方下载链接 &#xff0c;点击将直接转至苹果官网下载。❤️❤️❤️ Xcode官网&#xff1a;Xcode Releases | xcodereleases.com Xcode版本Xcode发布时间对应macOS版本macOS SDKsiOS SDKswatchOS SDKstvOS SDKs下载Xcode发…