Rust学习笔记:深度解析内存管理(二)

7f39db826ad89d2987bd97da1544abcd.jpeg

在这个信息爆炸的时代,学习一门新的编程语言不仅仅是为了找到一份好工作,更是为了打开思维的新窗口。Rust,作为一门注重安全、速度和并发的系统编程语言,正吸引着越来越多的年轻开发者的目光。今天,我们将一起深入探讨Rust的内存管理机制,包括它独特的所有权系统、借用规则以及引用/指针的使用,带你领略Rust语言的魅力所在。

在Rust中,内存管理是其核心特性之一,不同于其他语言需要开发者手动管理内存或完全依赖垃圾回收机制,Rust通过所有权、借用和生命周期等概念,有效防止了内存泄露和数据竞争等问题,确保了代码的安全性和高效性。

a189ce73f1cfdf17420c497928aceaf0.jpeg

内存管理入门:从传统到Rust的革新之路

在软件开发的世界里,如何高效、安全地管理内存是每个开发者都必须面对的挑战。不同的编程语言采取了不同的策略来解决这一问题,而Rust语言在这方面采用了一种独特且革命性的方法——所有权系统。在深入了解Rust的所有权之前,让我们先回顾一下其他语言是如何管理内存的。

传统内存管理方法

  • 垃圾收集(Garbage Collection):Java、Go等语言通过垃圾收集器自动查找并释放不再使用的内存。这种方法虽然减轻了开发者的负担,但可能会对性能产生不利影响。

  • 手动内存管理:C/C++等语言要求程序员手动分配和释放内存。这增加了程序的灵活性,但同时也增加了内存安全风险,需要开发者承担更多的责任。

  • 引用计数:Python等语言使用引用计数来跟踪每个对象的引用数量。当对象的引用计数降至零时,该对象被视为不再需要,并由垃圾收集器回收。

Rust的革新之路:所有权系统

Rust采用了一种全新的内存管理模型——所有权系统,它通过在编译时检查规则,并定义运行时行为来决定何时释放内存,从而实现了内存安全和性能的平衡。Rust的所有权系统基于三条基本规则:

  • Rust中的每个值都有一个所有者。

  • 一次只能有一个所有者拥有该值。

  • 当所有者离开作用域时,这个值会被自动释放。

这种方法不仅提高了内存安全性,还通过将大部分内存处理功能的检查放在编译时,提高了程序的性能。与传统的内存管理方法相比,Rust的所有权机制无疑提供了一种更为高效和安全的解决方案。

Rust内存管理:所有权与作用域

在Rust的学习之路上,理解内存管理是一道不可或缺的关卡。Rust通过所有权(Ownership)机制来管理内存,这一机制的核心在于:内存的每一块资源只能有一个所有者,当所有者结束生命周期时,相关资源将被自动释放。这听起来可能有些抽象,但通过几个简单的例子,我们可以更深入地理解这一概念。

所有权与变量作用域

让我们从最基本的例子开始:

fn main() {let s = String::from("Brian");println!("{}", s);
}

在这个例子中,当变量s被声明时,它在堆上分配了内存。根据Rust的规则,当拥有该内存的变量s离开作用域后,Rust会自动释放这部分内存。在这个例子里,变量s在main函数执行完毕后离开作用域。

再来看一个稍微复杂一点的例子,引入了内部作用域:

fn main() {{let s = String::from("Brian");println!("{}", s);// 内部作用域}let s2 = String::from("Brian 2");println!("{}", s2); // 外部作用域
}

在这里,由于增加了额外的大括号,s的作用域被限制在了内部大括号里,因此,当内部作用域结束时,s所占用的内存就会被释放。然后,外部作用域的s2同样在main函数结束时被释放。

所有权转移与克隆

Rust中的所有权机制确保了内存的安全使用,但这也意味着一块内存的所有权在任一时刻只能属于一个变量。看看下面这个例子:

fn main() {let s = String::from("Brian");let s2 = s;println!("{}", s); // 编译错误,因为s的所有权已经转移给了s2
}

要解决这个问题,我们可以使用克隆:

fn main() {let s = String::from("Brian");let s2 = s.clone();println!("{} : {}", s, s2); // 正常工作,因为s被克隆,所有权没有被转移
}

在Rust中,所有权(Ownership)是其内存管理的核心概念,通过一系列规则确保内存安全和程序效率。理解所有权的转移和借用是掌握Rust的关键。以下是对上述内容的补充和详细解释:

所有权的转移

  • 通过赋值或变量绑定改变所有权:当一个变量赋值给另一个变量时,原始变量的所有权会转移给新变量。这意味着之前的变量将无法再被访问,从而防止了悬垂指针或重复释放内存的问题。

  • 通过函数传递数据改变所有权:将变量作为参数传递给函数或从函数返回值时,所有权也可能发生转移。如果函数取得了某个值的所有权,那么原始变量将无法再次使用,除非这个值被返回。

防止问题的策略

为了避免由于所有权系统导致的使用限制,Rust提供了一些策略:

  • 使用引用:当不需要完全拥有值时,可以使用引用(&T和&mut T)来借用值。这样可以在不转移所有权的情况下访问或修改数据,同时保持内存安全。

  • 复制值:如果类型实现了Copy trait,那么在赋值或函数传递时,原始数据将被自动复制,而不是移动所有权。这适用于一些简单的类型,如整数类型和布尔类型,但不适用于如String这样的需要堆分配的类型。

  • 减少长寿命对象数量:通过重构代码来减少需要长时间持有的对象,可以减少内存占用和复杂度,提高程序效率。

  • 包装数据类型:通过创建或使用结构体(Structs)等类型来包装数据,可以更有效地管理数据的所有权和借用,尤其是在处理复杂数据结构时。

避免双重释放错误

所有权的一个重要原因是避免双重释放错误(double free error)。如果允许多个变量拥有同一块内存的所有权,当这些变量被销毁时,相同的内存会被释放多次,导致程序崩溃或安全漏洞。Rust通过确保每块内存只有一个所有者来防止这种情况发生。

函数与所有权

在Rust中,将变量传递给函数时,可能会发生所有权的移动或复制,这取决于变量的类型:

fn main() {let s = String::from("Brian");print_string(s);// println!("{}", s);  这将失败,因为s的所有权已经移动到了函数中let i = 192;print_int(i);println!("{}", i); // 这可以工作,因为i是基本类型,其大小已知且在栈上分配
}fn print_string(s_in : String) {println!("{}", s_in);
}fn print_int(i_in : i32) {println!("{}", i_in);
}

在Rust中,处理堆上分配的值(如String类型)与处理栈上分配的基本类型值(如i32)时,所有权的规则表现出明显的不同。通过前面提到的例子,我们可以深入探讨这一差异及其对函数调用和返回值的影响。

堆上分配的值与所有权

当我们调用print_string函数并传递一个String类型的变量时,这个变量的所有权被移动到了函数内部。因此,一旦函数调用完成,原始变量s就不再持有这个字符串的所有权,也就无法再次访问它。这是因为String类型的数据存储在堆上,Rust通过所有权机制来管理堆内存,确保内存安全。

栈上分配的基本类型与所有权

相比之下,基本类型如i32存储在栈上,当它们被传递给函数时,Rust会进行数据的拷贝而不是移动所有权。这意味着即使在调用print_int函数后,原始变量i仍然可以被访问,因为它的值在函数调用时被复制了。

函数返回值与所有权的转移

为了解决因所有权转移而导致的变量不可用的问题,我们可以通过函数返回值来重新获得所有权。在修改后的print_string例子中,函数接收一个String类型的参数,并将这个参数作为返回值返回。这样做的结果是,函数内部的所有权操作完成后,将所有权返回给调用者。

fn main() {let s = String::from("Brian");let s = print_string(s); // 将s的所有权传给函数,然后通过返回值重新获得所有权println!("{}", s); // 这里可以正常使用s,因为所有权已经通过函数返回值返回
}fn print_string(s_in: String) -> String {println!("{}", s_in);s_in // 返回s_in,这将所有权从函数内部转移回调用者
}

阴影(shadowing)与冻结变量

Rust还允许"阴影"变量,即在相同的作用域内用新的值重新声明同名变量:

fn main() {let shadowed_var = 12; {println!("before being shadowed: {}", shadowed_var);let shadowed_var = "abc"; println!("shadowed in inner block: {}", shadowed_var);}println!("outside inner block: {}", shadowed_var);let shadowed_var = 22; println!("shadowed in outer block: {}", shadowed_var);
}

最后,变量还可以被冻结,即在某个作用域内,之前可变的变量变为不可变:

fn main() {let mut mutable_var = 7i32;{let mutable_var = mutable_var;println!("{}", mutable_var);// mutable_var = 50; // 错误!在这个作用域内`mutable_var`是不可变的}mutable_var = 3;println!("{}", mutable_var);
}

通过这些例子,我们可以看到,Rust通过所有权、作用域、变量阴影和冻结等机制,提供了一种既高效又安全的方式来管理内存。这些概念初看起来可能有些复杂,但一旦掌握,你将能够编写出更加安全和高效的Rust代码。

借用(Borrowing)和引用(References)

在Rust中,借用(Borrowing)和引用(References)是管理和访问数据的关键机制,而不需要获取数据的所有权。这使得在不改变原始数据所有权的情况下,安全地共享和操作数据成为可能。

不可变借用

通过不可变借用,你可以创建对变量的引用,这样就可以读取或使用数据,而无需修改它。看看下面的例子:

fn main() {let data = String::from("Brian");let reference_a = &data;let reference_b = &data;println!("Original data: {}", data); // 因为我们采用了引用,所以data被借用了,并且仍然可以访问println!("Reference a: {}", reference_a);println!("Reference b: {}", reference_b);
}

如果你尝试移除第三行中reference_b声明的&符号,改为let reference_b = data;,编译器将会报错。这是因为reference_a已经“借用”了data的值,而此时你又尝试将data的所有权移动到reference_b,这违反了Rust的内存安全规则。

借用检查器

编译器中负责检查这些规则的部分叫做借用检查器(borrow checker)。当它发现代码可能违反Rust的内存规则时,它会引发错误,阻止代码编译。这种在编译时期就发现内存问题的能力,对于保持运行时性能和安全性来说是一个巨大的优势。

函数中的不可变引用

在函数中使用不可变引用是非常常见的,这允许你传递数据给函数而不转移所有权:

fn combined_length(s1: &String, s2: &String) -> usize {s1.len() + s2.len()
}fn main() {let first = String::from("Brian");let second = String::from("Enochson");let total_length = combined_length(&first, &second);println!("The combined length of my two string is: {}", total_length);// 因为我们只传递了引用,所以变量在这里仍然可以使用println!("Second string: {}", second);
}

可变借用

Rust同样支持可变借用,但需要明确声明。这符合Rust的设计哲学,旨在避免给开发者带来意外的行为:

fn main() {let mut first = String::from("Brian");let mut_second = &mut first;mut_second.push_str(" Enochson");println!("Modified data via reference: {}", mut_second);// 注意:此时尝试直接访问first可能会引起编译错误,因为已经存在对first的可变引用
}

解引用

使用*符号对引用的变量进行解引用,这不是类型转换,而是指示编译器“跟随”引用到底层类型:

fn swap(a: &mut i32, b: &mut i32) {let temp_v = *a;*a = *b;*b = temp_v;
}fn main() {let mut a = 5;let mut b = 10;swap(&mut a, &mut b);println!("After swap a: {}, b: {}", a, b); 
}

原始指针

虽然在Rust中直接操作原始指针不常见,但在某些场景,尤其是库开发中可能会用到。这通常需要使用unsafe代码块,因为它允许绕过Rust的安全保证:

fn main() {let x = 5;let raw = &x as *const i32; // 将x的引用转换为原始指针let points_at = unsafe { *raw };println!("raw pointers value is {}", points_at);
}

这里使用unsafe关键字是因为解引用原始指针可能会导致未定义行为,Rust要求开发者在这种情况下明确表明自己的意图。通过这些机制,Rust在提供强大功能的同时,确保了代码的安全性和高效性。

结束

在这第二期中,我们深入探讨了Rust的内存管理概念,并通过代码示例来凸显每个要点。我们研究了基于所有权的Rust独特的内存模型。同时,也覆盖了阴影(Shadowing)、借用(Borrowing)、引用(References)以及指针(Pointers)等主题。有了这些基础知识,我们将在下一篇《Rust学习笔记》文章中探讨流程控制,并更深入地研究函数。

Rust的内存安全特性和所有权系统提供了一种高效且安全的方式来管理内存,避免了传统编程语言中常见的内存泄漏和数据竞争问题。通过不可变和可变借用,Rust能够在编译时检查数据竞争,从而在不牺牲性能的情况下,确保并发安全。此外,Rust通过引用和指针提供了灵活的数据访问方式,同时保持了代码的安全性。

理解和掌握这些概念对于编写高效、安全的Rust代码至关重要。随着我们对Rust更深入的探索,你将能够利用这些强大的特性来构建可靠和高性能的应用程序。

期待在接下来的文章中,我们将继续探索Rust的更多高级特性,包括流程控制和函数等。希望你能在学习Rust的过程中发现其独特的魅力,并将这些知识应用到实际的项目中去。

相关内容

Rust学习笔记:基础概念介绍(一)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/722232.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据结构与算法:堆排序和TOP-K问题

朋友们大家好,本节内容来到堆的应用:堆排序和topk问题 堆排序 1.堆排序的实现1.1排序 2.TOP-K问题3.向上调整建堆与向下调整建堆3.1对比两种方法的时间复杂度 我们在c语言中已经见到过几种排序,冒泡排序,快速排序(qsor…

微信小程序云开发教程——墨刀原型工具入门(安装以及基础使用教程)

引言 作为一个小白,小北要怎么在短时间内快速学会微信小程序原型设计? “时间紧,任务重”,这意味着学习时必须把握微信小程序原型设计中的重点、难点,而非面面俱到。 要在短时间内理解、掌握一个工具的使用&#xf…

稀碎从零算法笔记Day4-LeetCode:交替合并字符串

前言:今天妹有深夜档,因为8点有个飞机 题型:字符串、双指针(笔者没用这个思路) 链接:1768. 交替合并字符串 - 力扣(LeetCode) 来源:LeetCode 著作权归作者所有。商业转…

JasperStudio中TextField文本框组件渲染之后,出现行间距不一致的问题

目录 1.1、问题描述 1.2、解决方案 1.1、问题描述 最近在处理线上遇到的一个问题,是有关JasperReports报表相关的问题,问题背景大概是这样的:我们的项目中使用了JasperReports来渲染报表,其中使用到了Text Field文本框组件,但是问题是渲染出来的数据直接会出现一些间距…

洛谷:P3068 [USACO13JAN] Party Invitations S(枚举、前缀和)

这题我们数据范围太大,用二维肯定是不行的,我们可以采用一维线性存储。 如题意,我们可以将每组奶牛编号都存在一维数组里面,只需记录每组的头尾指针就可以了。 如题中样例我们就可以存储成1 3 3 4 1 2 3 4 5 6 7 4 3 2 1 然后第…

[LeetBook]【学习日记】寻找和为指定数字的连续数字

题目 文件组合 待传输文件被切分成多个部分,按照原排列顺序,每部分文件编号均为一个 正整数(至少含有两个文件)。传输要求为:连续文件编号总和为接收方指定数字 target 的所有文件。请返回所有符合该要求的文件传输组…

【kubernetes】关于k8s集群的存储卷

目录 一、存储卷的分类 二、empty存储卷以及特点 三、hostpath存储卷以及特点 四、nfs存储卷以及特点 五、pvc存储卷 查看pv的定义 查看pvc的定义 实操:静态创建pv的方式 实现pvc存储卷 步骤一:先完成nfs的目录共享,需要准备不同的目…

C# 中 TryParse 将字符串转换为特定类型的方法

在 C# 中,TryParse 是一个用于将字符串转换为特定类型的方法。它用于尝试解析字符串并将其转换为指定类型的值,而不会引发异常。如果解析成功,它将返回 true 并将解析结果存储在输出参数中;如果解析失败,它将返回 fals…

redis10 应用问题(穿透、击穿、雪崩、分布式锁)

思维草图 缓存穿透 查询不存在的数据,穿透redis缓存,请求直接攻击后端db。 问题 当系统中引入redis缓存后,一个请求进来后,会先从redis缓存中查询,缓存有就直接返回(相当于一道隔离闸,保护db…

vue设计原理-带你重走vue诞生路程

我们首先看下面这个小demo demo源码: <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" conten…

JAVA如何利用接口实现多继承问题

hello&#xff0c;上文带大家学习了java中类的继承&#xff0c;我们可以创建一个父类&#xff0c;将类中的共性抽取出来&#xff0c;通过子类继承的方式来实现代码的复用。今天带大家学习不同类之间的另外几种关系&#xff0c;即多态抽象类和接口。 多态的概念 多态&#xff0c…

Vue 3的Composition API和vue2的不同之处

Vue 3的Composition API是Vue.js框架的一个重要更新&#xff0c;它提供了一种新的组件逻辑组织和复用方式。在Vue 2中&#xff0c;我们通常使用Options API&#xff08;data、methods、computed等&#xff09;来组织组件的逻辑&#xff0c;但这种组织方式在处理复杂组件时可能会…

实体店好做还是电商平台好做?哪个发展前景较高?

我是电商珠珠 大部分人面对新的一年都会比较迷茫&#xff0c;想要创业会自己增加收入&#xff0c;却在开店和线上做电商这两者之间犹豫不决。不知道哪个更加合适自己&#xff0c;换句话来说&#xff0c;就是不知道哪个赔付率低一点。 现在这个行情&#xff0c;按照网友的说法…

C++ string类详解及模拟实现

目录 【本节目标】 1. 为什么学习string类&#xff1f; 1.1 C语言中的字符串 1.2 面试题(暂不做讲解) 2. 标准库中的string类 2.1 string类(了解) 2.2 string类的常用接口说明&#xff08;注意下面我只讲解最常用的接口&#xff09; 3. string类的模拟实现 3.1string类常用…

Express学习(一)

Express Express简介 什么是Express 官方给出的概念&#xff1a;Express是基于Node.js平台&#xff0c;快速、开放、极简的web开发框架。 通俗的理解&#xff1a;Express的作用和Node.js内置的http模块类似&#xff0c;是专门用来创建Web服务器的。进一步理解Express 不使用E…

【软件使用】Markdown编辑器第一次使用介绍

【软件使用】Markdown编辑器第一次使用介绍 markdown格式支持的软件有&#xff1a;VS Code 和 Typora&#xff0c;CSDN写网页博文也是用的.md&#xff0c;CSDN能支持导入的文件也是以.md格式结尾的文件名。 欢迎使用Markdown编辑器 你好&#xff01; 这是你第一次使用 Markd…

C语言——结构体(位段)、联合体、枚举

hello&#xff0c;大家好&#xff01;我是柚子&#xff0c;今天给大家分享的内容是C语言中的自定义类型结构体、联合体以及枚举&#xff0c;有什么疑问或建议可以在评论区留言&#xff0c;会顺评论区回访哦~ 一、结构体 struct a.结构体声明 不同于数组的是&#xff0c;结构…

【CSP试题回顾】202212-2-训练计划

CSP-202212-2-训练计划 解题思路 输入和初始化&#xff1a; 首先&#xff0c;代码从输入中获取项目的截止日期和项目数量。然后&#xff0c;它初始化一个项目列表&#xff0c;每个项目都有其依赖项、被依赖的项目集合、完成时间、总完成时间&#xff08;包括依赖链&#xff09…

【JavaSE】面向对象——多态性

多态性 多态性的概念 所谓多态性&#xff0c;理解为一个事物的多种形态。具体点就是去完成某个动作时&#xff0c;不同的对象会产生不同的状态。 多态性的好处 多态在Java中指的是父类的引用指向子类的对象&#xff0c;或者可以说是子类的对象赋给父类的引用。这样在我们的…

SpringBoot实现分页模糊查询

1. Navicat查询数据 Navicat中查询所有数据 SELECT * FROM sys_user;Navicat中查询前两条数据&#xff08;俩种方式&#xff09; SELECT * FROM sys_user LIMIT 2; //从0开始&#xff0c;第一个参数是起始位置即(pageNum-1)*pageSize&#xff0c;第二个参数是步长 SELECT * …