Rust 所有权特性详解

Rust 所有权特性详解

Rust 的所有权系统是其内存安全的核心机制之一。通过所有权规则,Rust 在编译时避免了常见的内存错误(如空指针、数据竞争等)。本文将从堆内存与栈内存、所有权规则、变量作用域、String 类型、内存分配、所有权移动、Clone、栈内存的 Copy、所有权与函数、返回值与作用域等角度详细介绍 Rust 的所有权特性,并通过综合示例展示这些知识点的实际应用。


1. 什么是堆内存和栈内存

  • 栈内存

    • 后进先出(LIFO)的数据结构。
    • 分配和释放速度快。
    • 用于存储固定大小的数据(如基本类型,Rust的基本类型有哪些,他们存在堆内存还是栈内存?)。
  • 堆内存

    • 动态分配的内存区域。
    • 分配和释放速度较慢。
    • 用于存储大小可变或生命周期不确定的数据(如 StringVec)。

示例:栈内存与堆内存

fn main() {
    let x = 5; // x 存储在栈上
    let s = String::from("你好"); // s 的数据存储在堆上,指针存储在栈上

    println!("x: {}, s: {}", x, s);
}

输出

x: 5, s: 你好

分析

  • x 是基本类型,存储在栈上。
  • sString 类型,数据存储在堆上,指针和长度等信息存储在栈上。

2. Rust 所有权的规则

Rust 的所有权规则如下:

  1. 每个值都有一个所有者。
  2. 同一时间只能有一个所有者。
  3. 当所有者离开作用域时,值会被自动释放。

示例:所有权规则

fn main() {
    let s1 = String::from("你好");
    let s2 = s1; // s1 的所有权转移到 s2

    // println!("{}", s1); // 错误:s1 不再拥有数据
    println!("s2: {}", s2);
}

输出

s2: 你好

分析

  • s1 的所有权在赋值给 s2 后转移,s1 不再有效。

3. 变量的作用域

变量的作用域是从声明开始到当前块结束。

示例:变量作用域

fn main() {
    let s = String::from("你好"); // s 进入作用域

    {
        let inner_s = String::from("内部"); // inner_s 进入作用域
        println!("内部作用域: {}", inner_s);
    } // inner_s 离开作用域,内存被释放

    println!("外部作用域: {}", s);
} // s 离开作用域,内存被释放

输出

内部作用域: 内部
外部作用域: 你好

分析

  • inner_s 的作用域仅限于内部块。
  • s 的作用域是整个 main 函数。

4. String 类型

String 是 Rust 中动态分配的字符串类型,存储在堆上。

示例:String 类型

fn main() {
    let mut s = String::from("你好");
    s.push_str(", Rust!"); // 修改字符串
    println!("{}", s);
}

输出

你好, Rust!

分析

  • String 类型允许动态修改内容。

5. 内存分配

Rust 通过所有权系统自动管理堆内存的分配和释放。

示例:内存分配

fn main() {
    let s = String::from("你好"); // 分配堆内存
    println!("{}", s);
} // s 离开作用域,内存被释放

输出

你好

分析

  • String::from 分配堆内存。
  • s 离开作用域时,内存被自动释放。

6. 所有权移动时变量和数据的状态变化

当所有权从一个变量移动到另一个变量时,原始变量将失效。

示例:所有权移动

fn main() {
    let s1 = String::from("hello");
    let s2 = s1; // s1 的所有权移动到 s2

    // println!("{}", s1); // 错误:s1 不再有效
    println!("s2: {}", s2);
}

输出

s2: hello

分析
在这里插入图片描述
-s1的指针存在栈内存,栈内存的value指向堆内存的第一个索引位置。
在这里插入图片描述

  • 当执行s2=s1的时候,仅仅复制了栈内存上的数据,堆内存的内容不不变。如果堆内存上的数据非常大,复制的操作成本会无限增加!
  • Double Free问题:当前s1、s2都指向同一份数据,当这两个变量离开作用域时,他们会同时释放同一块内存,这就会引起Double Free安全问题。为了确保内存安全,当执行到语句let s2=s1时,Rust让s1失效,也称之为将所有权转移给了s2
  • s1 的所有权转移给 s2 后,s1 失效(如下图所示)。
    在这里插入图片描述

7. 作用域和内存分配

变量的作用域决定了其内存的生命周期。

示例:作用域和内存分配

fn main() {
    let s = String::from("你好"); // s 进入作用域,分配内存
    println!("{}", s);
} // s 离开作用域,内存被释放

输出

你好

分析

  • s 的作用域结束后,内存被自动释放。

8. Clone

Clone 允许显式复制堆上的数据。

示例:Clone

fn main() {
    let s1 = String::from("你好");
    let s2 = s1.clone(); // 显式复制数据

    println!("s1: {}, s2: {}", s1, s2);
}

输出

s1: 你好, s2: 你好

分析

  • clone 会复制堆上的数据,s1s2 都有效。

9. 栈内存的 Copy

基本类型实现了 Copy trait,赋值时会复制值而不是移动所有权。

示例:栈内存的 Copy

fn main() {
    let x = 5;
    let y = x; // x 的值被复制

    println!("x: {}, y: {}", x, y);
}

输出

x: 5, y: 5

分析

  • xy 都有效,因为 i32 实现了 Copy

那么,哪些类型实现了 Copy 特质呢?你可以查看特定类型的文档来确认,但一般来说,任何由简单标量值组成的类型都可以实现 Copy,而任何需要分配内存或是某种形式的资源的类型则不能实现 Copy。以下是一些实现了 Copy 的类型:

  • 所有的整数类型,例如 u32
  • 布尔类型 bool,其值为 truefalse
  • 所有的浮点数类型,例如 f64
  • 字符类型 char
  • 元组,如果它们只包含同样实现了 Copy 的类型。例如,(i32, i32) 实现了 Copy,但 (i32, String) 则没有。

10. 所有权和函数

将值传递给函数会转移所有权。

示例:所有权和函数

fn take_ownership(s: String) {
    println!("函数内部: {}", s);
} // s 离开作用域,内存被释放

fn main() {
    let s = String::from("你好");
    take_ownership(s); // s 的所有权转移到函数

    // println!("{}", s); // 错误:s 不再有效
}

输出

函数内部: 你好

分析

  • s 的所有权在传递给函数后转移。

11. 返回值和作用域

函数可以通过返回值转移所有权。

示例:返回值和作用域

fn give_ownership() -> String {
    let s = String::from("你好");
    s // 返回 s,所有权转移给调用者
}

fn main() {
    let s = give_ownership(); // s 获得所有权
    println!("{}", s);
}

输出

你好

分析

  • give_ownership 返回 s,所有权转移给 main 函数中的 s

综合示例

以下是一个综合示例,展示了所有权、作用域、CloneCopy、函数与返回值的用法:

fn main() {
    // 栈内存的 Copy
    let x = 5;
    let y = x; // x 的值被复制
    println!("x: {}, y: {}", x, y);

    // 堆内存的所有权
    let s1 = String::from("你好");
    let s2 = s1.clone(); // 显式复制数据
    println!("s1: {}, s2: {}", s1, s2);

    // 所有权和函数
    let s3 = String::from("世界");
    take_ownership(s3); // s3 的所有权转移到函数
    // println!("{}", s3); // 错误:s3 不再有效

    // 返回值和作用域
    let s4 = give_ownership(); // s4 获得所有权
    println!("s4: {}", s4);
}

fn take_ownership(s: String) {
    println!("函数内部: {}", s);
} // s 离开作用域,内存被释放

fn give_ownership() -> String {
    let s = String::from("你好,世界");
    s // 返回 s,所有权转移给调用者
}

输出

x: 5, y: 5
s1: 你好, s2: 你好
函数内部: 世界
s4: 你好,世界

分析

  1. xy 是基本类型,赋值时复制值。
  2. s1s2String 类型,使用 clone 显式复制数据。
  3. s3 的所有权在传递给函数后转移。
  4. s4 通过函数返回值获得所有权。

总结

Rust 的所有权系统通过以下特性确保内存安全:

  1. 堆内存与栈内存:区分数据的存储位置。
  2. 所有权规则:确保每个值只有一个所有者。
  3. 作用域:决定变量的生命周期。
  4. String 类型:动态分配的字符串。
  5. 内存分配:自动管理堆内存。
  6. 所有权移动:转移所有权时原始变量失效。
  7. Clone:显式复制堆数据。
  8. 栈内存的 Copy:基本类型赋值时复制值。
  9. 所有权与函数:传递值会转移所有权。
  10. 返回值与作用域:通过返回值转移所有权。

通过合理使用这些特性,可以编写出高效且安全的 Rust 代码。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于深度学习的视觉检测小项目(十七) 用户管理后台的编程

完成了用户管理功能的阶段。下一阶段进入AI功能相关。所有的资源见文章链接。 补充完后台代码的用户管理界面代码: import sqlite3from PySide6.QtCore import Slot from PySide6.QtWidgets import QDialog, QMessageBoxfrom . import user_manage # 导入使用ui…

ubuntuCUDA安装

系列文章目录 移动硬盘制作Ubuntu系统盘 前言 根据前篇“移动硬盘制作Ubuntu系统盘”安装系统后,还不能够使用显卡。 如果需要使用显卡,还需要进行相关驱动的安装(如使用的为Nvidia显卡,就需要安装相关的Nvidia显卡驱动&#xff…

Docker Compose的使用

文章首发于我的博客:https://blog.liuzijian.com/post/docker-compose.html 目录 Docker Compose是什么Docker Compose安装Docker Compose文件Docker Compose常用命令案例:部署WordPress博客系统 Docker Compose是什么 Docker Compose是Docker官方的开源…

【深度分析】DeepSeek 遭暴力破解,攻击 IP 均来自美国,造成影响有多大?有哪些好的防御措施?

技术铁幕下的暗战:当算力博弈演变为代码战争 一场针对中国AI独角兽的全球首例国家级密码爆破,揭开了数字时代技术博弈的残酷真相。DeepSeek服务器日志中持续跳动的美国IP地址,不仅是网络攻击的地理坐标,更是技术霸权对新兴挑战者的…

使用python实现与本地ollama部署的deepseek对话

专栏总目录 按照ollama官方doc的example操作,没有成功与本地ollama上的deepseek-r1:1.5b通讯后,发现vscode可以调用本地ollama上的deepseek模型。 为了实现与ollama上的deepseek模型通讯,我使用wireshark对本地回环地址进行侦听后&#xff0c…

基于STM32的智能健康监测手环

1. 引言 随着可穿戴设备的普及,健康监测技术正逐步融入日常生活。本文设计了一款基于STM32的智能健康监测手环,能够实时采集用户心率、血氧饱和度、体温及运动数据,并通过低功耗蓝牙(BLE)与手机APP交互。该系统结合了…

OpenGL学习笔记(六):Transformations 变换(变换矩阵、坐标系统、GLM库应用)

文章目录 向量变换使用GLM变换(缩放、旋转、位移)将变换矩阵传递给着色器坐标系统与MVP矩阵三维变换绘制3D立方体 & 深度测试(Z-buffer)练习1——更多立方体 现在我们已经知道了如何创建一个物体、着色、加入纹理。但它们都还…

麦芯(MachCore)应用开发教程5 --- 工位和晶圆传输

麦芯是构建在windows系统上的设备应用操作系统,利用该系统可以快速高效的开发一款设备专用软件。希望进一步了解请email: acloud163.com 黄国强 2025/02/03 一、工位与子设备的关系 想象工厂中的流水线工作站,每个工位(Station&#xff09…

冰蝎v3.0 beta7来啦

我用了一台kali,一台centos,一台windows,做了一个文件上传和一个反弹shell实验,载荷是AES加密的,终于感受到了对加密流量的无可奈何~ kali(php8.1)centos(php7.1)window…

Golang 并发机制-5:详解syn包同步原语

并发性是现代软件开发的一个基本方面,Go(也称为Golang)为并发编程提供了一组健壮的工具。Go语言中用于管理并发性的重要包之一是“sync”包。在本文中,我们将概述“sync”包,并深入研究其最重要的同步原语之一&#xf…

【PyQt】超级超级笨的pyqt计算器案例

计算器 1.QT Designer设计外观 1.pushButton2.textEdit3.groupBox4.布局设计 2.加载ui文件 导入模块: sys:用于处理命令行参数。 QApplication:PyQt5 应用程序类。 QWidget:窗口基类。 uic:用于加载 .ui 文件。…

Flutter Scaffold 页面结构

Material是一套设计风格,提供了大量的小部件,这里用Material风格搭建一个常见的应用页面结构。 创建Material应用 import package:flutter/material.dart;class App extends StatelessWidget {overrideWidget build(BuildContext context) {return Mat…

2月3日星期一今日早报简报微语报早读

2月3日星期一,农历正月初六,早报#微语早读。 1、多个景区发布公告:售票数量已达上限,请游客合理安排行程; 2、2025春节档总票房破70亿,《哪吒之魔童闹海》破31亿; 3、美宣布对中国商品加征10…

大模型本地化部署(Ollama + Open-WebUI)

文章目录 环境准备下载Ollama模型下载下载Open-WebUI 本地化部署的Web图形化界面本地模型联网查询安装 Docker安装 SearXNG本地模型联网查询 环境准备 下载Ollama 下载地址:Ollama网址 安装完成后,命令行里执行命令 ollama -v查看是否安装成功。安装成…

10 Flink CDC

10 Flink CDC 1. CDC是什么2. CDC 的种类3. 传统CDC与Flink CDC对比4. Flink-CDC 案例5. Flink SQL 方式的案例 1. CDC是什么 CDC 是 Change Data Capture(变更数据获取)的简称。核心思想是,监测并捕获数据库的变动(包括数据或数…

【25考研】南开软件考研复试复习重点!

一、复试内容 复试采取现场复试的方式。复试分为笔试、机试和面试三部分。三部分合计100分,其中笔试成绩占30%、机试成绩占30%、面试成绩占40%。 1.笔试:专业综合基础测试 考核方式:闭卷考试,时长为90分钟。 笔试考查内容范围…

【Cadence仿真技巧学习笔记】求解65nm库晶体管参数un, e0, Cox

在设计放大器的第一步就是确定好晶体管参数和直流工作点的选取。通过阅读文献,我了解到L波段低噪声放大器的mos器件最优宽度计算公式为 W o p t . p 3 2 1 ω L C o x R s Q s p W_{opt.p}\frac{3}{2}\frac{1}{\omega LC_{ox}R_{s}Q_{sp}} Wopt.p​23​ωLCox​Rs…

【leetcode练习·二叉树拓展】归并排序详解及应用

本文参考labuladong算法笔记[拓展:归并排序详解及应用 | labuladong 的算法笔记] “归并排序就是二叉树的后序遍历”——labuladong 就说归并排序吧,如果给你看代码,让你脑补一下归并排序的过程,你脑子里会出现什么场景&#xff…

[ESP32:Vscode+PlatformIO]新建工程 常用配置与设置

2025-1-29 一、新建工程 选择一个要创建工程文件夹的地方,在空白处鼠标右键选择通过Code打开 打开Vscode,点击platformIO图标,选择PIO Home下的open,最后点击new project 按照下图进行设置 第一个是工程文件夹的名称 第二个是…

Docker 部署教程jenkins

Docker 部署 jenkins 教程 Jenkins 官方网站 Jenkins 是一个开源的自动化服务器,主要用于持续集成(CI)和持续交付(CD)过程。它帮助开发人员自动化构建、测试和部署应用程序,显著提高软件开发的效率和质量…