Rust 所有权

所有权

  • Rust的核心特性就是所有权
  • 所有程序在运行时都必须管理他们使用计算机内存的方式
    • 有些语言有垃圾收集机制,在程序运行时,他们会不断地寻找不再使用的内存
    • 在其他语言中,程序员必须显式的分配和释放内存
  • Rust采用了第三种方式:
    • 内存是通过一个所有权系统来管理的,其中包含一组编译器在编译时检查的规则
    • 在程序运行时,所有权特性不会减慢程序的运行速度。

Stack VS Heap

  • 在像Rust这样的系统级编程语言里,一个值是在还是在heap上对语言的行为和你为什么要做某些决定是有更大的影响的
  • 在你的代码运行的时时候,Stack和Heap都是你可用的内存,但他们的结构很不相同

存储数据

  • Stack按值的接收顺序来存储,按相反的顺序将他们移除(后进先出)
    • 添加数据叫压入栈
    • 移除数据叫弹出栈
  • 所有存储在stack上的数据必须拥有已知的固定的大小
    • 编译时大小位置的数据或运行时大小可能发生变化的数据必须存放在heap上
  • Heap内存组织性差一些
    • 当你把数据放入heap时,你会请求一定数量的空间
    • 操作系统在heap里找到一块足够大的空间,把它标记为在用,并返回一个指针,也就是这个空间的地址
    • 这个过程叫做在heap上进行分配,有时仅仅称为"分配"
  • 把值压到stack上不叫分配
  • 因为指针是已知固定大小的,可以把指针放在stack上
    • 但如果想要实际数据,你必须使用指针来定位
  • 把数据压到stack上比在heap上分配快得多
    • 因为操作系统不需要寻找用来存储新数据的空间,那个位置永远都是在stack的顶端
  • 在heap上分配空间需要做更多的工作
    • 操作系统首先需要找到一个足够大的空间来存放数据,然后要做号记录方便下次分配

访问数据

  • 访问heap中的数据要比访问stack中的数据慢,因为需要通过指针才能找到heap中的数据
    • 对于现代的处理器来说,由于缓存的缘故,如果指针在内存中跳转的次数越少,那么速度就越快
  • 如果数据存放的距离比较近,那么处理器的处理速度就会快一些(stack上)
  • 如果数据之间存放的距离比较远,那么处理速度就会慢一些(heap上)
    • 在heap上分配大量的空间也是需要时间的

函数调用

  • 当你的代码调用函数时,值被传入函数(也包括指向heap的指针)。函数本地的变量被压到stack上。当函数结束后,这些值会从stack上弹出

所有权存在的原因

  • 所有权存在的原因
    • 追踪代码的那些部分正在使用heap的哪些数据
    • 最小化heap上的重复数据量
    • 清理heap上未使用的数据以避免空间不足

所有权规则

  • 每个值都有一个变量,这个变量是该值的所有者
  • 每个值同时只能有一个所有者
  • 当所有者超出作用域(scope)时,该值将被删除

变量作用域

  • Scope就是程序中一个项目的有效范围
fn main(){// s不可用let s = "hello"; // s可用// 可以对s进行相关操作
}// s作用域到此结束,s不再可用

String类型

  • String比那些基础标量数据类型更复杂
  • 字符串字面值:程序里手写的那些字符串值。他们是不可变的
  • Rust还有第二种字符串类型:String。
    • 在heap上分配。能够存储在编译时未知数量的文本

创建String类型的值

  • 可以使用from函数从字符串字面值创建出String类型
fn main(){let mut s = String::from("Hello"); // ::表示from是String类型下的函数s.push_str(",world");println!("{}",s);
}
  • 这类字符串是可以被修改的
    • 因为他们处理内存的方式不同,

内存和分配

  • 字符串字面值,在编译时就知道他的内容了,其文本内容直接被硬编码到最终的可执行文件里
    • 速度快,高效。是因为其不可变性
  • String类型:为了支持可变性,需要在heap上分配内存来保存编译时未知的文本内容:
    • 操作系统必须在运行时来请求内存
      • 这步通过调用String::from来实现
    • 当用完String后,需要使用某种方式将内存返回给操作系统
      • 这步,在拥有GC的语言中,GC会跟踪并清理不再使用的内存
      • 没有GC就需要我们识别内存何时不再使用,并调用代码将他返回
        • 如果忘了,就会浪费内存
        • 如果提前做了,变量就会非法
        • 如果做了两次,也是Bug。必须一次分配对应一次释放
  • Rust采用了不同的方式:对于某个值来说,当拥有它的变量走出作用范围时,内存会立即自动的交换给操作系统。

变量和数据交互的方式:移动(Move)

  • 多个变量可以与同一个数据使用一种独特的方式来交互
fn main(){let x = 5;let y = x;// 整数是已知且固定大小的简单的值,这两个5被压到了stack中
}

String版本

fn main(){let s1 = String::from("hello");let s2 = s1;
}
  • 一个String由三部分组成
    • 一个指向存放字符串内容的内存的指针
    • 一个长度
    • 一个容量
  • 上面这些东西放在stack上
  • 存放字符串内容的部分在heap上
  • 长度len,就是存放字符串内容所需的字节数
  • 容量capacity是指String从操作系统总共获得的内存的总字节数

在这里插入图片描述

  • 当把s1赋给s2,String的数据被复制了一份
    • 在stack上复制了一份指针,长度,容量
    • 并没有复制指针所指向的heap上的数据
  • 当变量离开作用域时,Rust会自动调用drop函数,并将变量使用的heap内促释放
  • 当s1,s2离开作用域时,他们都会尝试释放相同的内存
    • 二次释放(double free)bug
  • 为了保证内存安全
    • Rust没有尝试复制被分配的内存
    • Rust让s1失效
      • 当s1离开作用域的时候,Rust不需要释放任何东西
fn main(){let s1 = String::from("hello");let s2 = s1;// 此时我们去使用s1println!("{}",s1);
}

会报错

在这里插入图片描述

  • 浅拷贝(shallow copy)
  • 深拷贝(deep copy)
  • 你也许会将复制指针,长度,容量视为浅拷贝,但由于Rust让s1失效了,所以我们用一个新的术语:移动(Move)
  • 隐含的一个设计原则:Rust不会自动创建数据的深拷贝
    • 就运行时性能而言,任何自动赋值的操作都是廉价的

变量和数据交互的方式:克隆(Clone)

  • 如果真想对heap上面的String数据进行深度拷贝,而不仅仅是stack上的数据,可以使用clone方法
fn main(){let s1 = String::from("hello");let s2 = s1.clone();println!!("{},{}",s1,s2);
}

在这里插入图片描述

Stack上的数据:复制

fn main(){let x = 5;let y = x;println!("{},{}",x,y);
}
  • Copy trait,可以用于像整数这样完全存放在stack上面的类型
  • 如果一个类型实现了Copy这个trait,那么旧的变量在赋值后仍然可用
  • 如果一个类型或者该类型的一部分实现了drop trait,那么Rust不允许让他再去实现Copy trait了

一些拥有Copy trait的类型

  • 任何简单标量的组合类型都可以是Copy的
  • 任何需要分配内存或某种资源的都不是Copy的
  • 一些拥有Copy trait的类型:
    • 所有的整数类型,例如u32
    • bool
    • char
    • 所有的浮点类型,例如f64
    • tuple(元组),如果其所有的字段都是Copy的

所有权与函数

  • 在语义上,将值传递给函数和把值赋给变量是类似的:

    • 将值传递给函数将发生移动复制
    fn main(){let s = String::from("Hello World");take_ownership(s); // s失效,因为函数结束后会调用drop方法let x = 5;makes_copy(x); // 因为分配在stack中,无事发生println!("x:{}",x)
    }fn take_ownership(some_string:String){println!("{}",some_string);
    }fn makes_copy(some_number:i32){println!("{}",some_number);
    }
    

返回值与作用域

  • 函数在返回值的过程中同样也会发生所有权的转移
fn main() {let s1 = gives_ownership();let s2 = String::from("hello");let s3 = takes_and_gives_back(s2); 
}fn gives_ownership() -> String {let some_string = String::from("hello");some_string
}fn takes_and_gives_back(a_string:String) -> String {a_string
}
  • 一个变量的所有权总是遵循同样的模式
    • 把一个值赋给其他变量时就会发生移动
    • 当一个包含了heap数据的变量离开作用域时,他的值就会被drop函数清理,除非数据的所有权移动到另一个变量上了

如何让函数使用某个值,但不获得其所有权

fn main(){let s1 = String::from("hello");let (s2,len) = calculate_length(s1);println!("the length of '{}' is {}",s2,len);
}fn calculate_length(s:String) -> (String,usize){let length = s.len();(s,length)
}
  • Rust有一个特性叫做"引用(Reference)"

引用和借用

fn main(){let s1 = String::from("hello");let len = calculate_length(s1);println!("the length of '{}' is {}",s1,len);
}fn calculate_length(s:&String) ->usize {s.len();
}
  • 参数的类型是&String而不是String
  • &符号就表示引用:允许你引用某些值而不取得其所有权
  • 我们把引用作为函数参数的这个行为叫做借用
  • 是否可以修改借用的东西?
    • 不行
  • 和变量一样,引用默认也是不可变的

可变引用

fn main(){let mut s1 = String::from("Hello");let len = calculate_length(&mut s1);println!("the length of '{}' is {}",s1,len);
}fn calculate_length(s:&mut String) -> usize {s.push_str(",world");s.len()
}
  • 可变引用有一个重要的限制:在特定作用域内,对某一块数据,只能有一个可变的引用
    • 这样做的好处是可在编译时防止数据竞争
  • 以下三种行为下会发生数据竞争:
    • 两个或多个指针同时访问同一个数据
    • 至少有一个指针用于写入数据
    • 没有使用任何机制来同步对数据的访问
  • 我们可以通过创建新的作用域,来允许非同时的创建多个可变引用
fn main(){let mut s = String::from("Hello");{let s1 = &mut s;}let s2 = &mut s;
}
  • 不可以同时拥有一个可变引用和不可变引用
  • 多个不可变的引用是可以的

悬空引用(Dangling References)

  • 悬空指针(Dangling Pointer):一个指针引用了内存中的某个地址,而这块内存可能已经释放并分配给其它人使用了
  • 在Rust里,编译器可保证引用永远都不是悬空引用。
    • 如果你引用了某些数据,编译器将保证在引用离开作用域之前数据都不会离开作用域

引用的规则

  • 在任何给定的时刻,只能满足下列条件之一
    • 一个可变的引用
    • 任意数量不可变的引用
  • 引用必须一直有效

切片

  • Rust的另外一种不持有所有权的数据类型:切片(Slice)
  • 字符串切片是指向字符串中一部分内容的引用
  • 形式:[开始索引…结束索引] (左闭右开)

注意

  • 字符串切片的索引范围必须发生在有效的UTF-8字符边界内
  • 如果尝试从一个多字节的字符中创建字符串切片,程序会报错并退出

将字符串切片作为参数传递

  • 采用&str作为参数类型,这样就可以同时接受String和&str类型的参数了
  • 定义函数时使用字符串切片来代替字符串引用会使我们的API更加通用,且不会损失任何功能。

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

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

相关文章

鲁班上门维修安装系统源码开发之功能模式

鲁班上门维修安装系统在当今的趋势呈现出显著的增长与创新。随着物联网、智能家居的普及,以及消费者对便捷、高效生活方式的追求,鲁班上门维修安装系统凭借其多渠道预约、智能派单、在线支付与费用明细透明等优势,赢得了市场的广泛认可。 …

【13.PIE-Engine案例——加载Landsat8 Collection2 SR数据集】

原始链接 原始路径欢迎大家登录航天宏图官网查看本案例原始来源 结果展示 具体代码 /*** File : Landsat8 Collection2 SR* Time : 2021/5/24* Author : piesat* Version : 1.0* Contact : 400-890-0662* License : (C)Copyright 航天宏图信息技术股份有…

【mathtype】word中如何输入4×4的矩阵,甚至阶数更多

在写论文或者使用word操作的时候,我们可能会使用矩阵插入我们所写的word中,今天小编就分享一下如何在word中输入矩阵。首先,我们word中需要安装mathtype的插件。 ①打开word,鼠标点击mathtype,再点击内联 ② 出现以下…

大模型常见的问题

什么是涌现现象 即模型在没有被明确训练执行某些任务的情况下,却能够展现出完成这些任务的能力。这是 因为模型在处理大量数据时学习到了复杂的模式和结构,从而能够泛化到未见过的任务上。 LLM的结构是什么样的 大语言模型通常基于Transformer架构&#…

读零信任网络:在不可信网络中构建安全系统10认证身份

1. 用户所知道的信息 1.1. 只有用户本人知道的信息 1.2. 密码 1.2.1. 密码是常用的认证机制 1.2.2. 密码验证就是确认用户“所知”性的较好途径 1.2.3. 用户可以利用密码管理器来便捷地管理多个高强度密码,从而有效降低数据泄露风险 1.2.4. 长度足够长 1.2.4.1…

数据结构——优先队列

文章目录 一、基本介绍二、基本操作三、实现1 实现的思路2 大顶堆实现2.1 概念2.2 完全二叉树的实现方式2.3 优先队列的图示2.4 对于基本操作实现的讲解2.4.1 检查队列是否为空 ( isEmpty )2.4.2 检查队列是否已满 ( isFull )2.4.3 查看 ( peek )2.4.4 插入 ( offer )2.4.5 删除…

本地GitLab runner自动编译Airoha项目

0 Preface/Foreword 1 GitLab runner环境 具体情况如下: Gitlab-ruuner运行在wsl 1中的Ubuntu 18.04 distro上专门为GitLab-runner分配了一个用户,名为gitlab-runner 2 自动编译 2.1 Permission denied 编译过程中,有两个文件出现权限不允…

基于风险的完整性和检查建模(RBIIM)MATLAB仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.本算法原理 4.1 Prior Density (先验密度) 4.2 Posterior Perfect Inspection (后验完美检验) 4.3 Posterior Imperfect Inspection (后验不完美检验) 4.4Cumulative Posterior Imperfect Inspection…

嵌入式安全:Provencore Secure os

嵌入式安全有何独特之处? 嵌入式安全领域的领导者 ProvenRun 宣布,其旗舰产品 ProvenCore for ARM™ Cortex-A 最近获得了 通用标准 (CC) EAL7 认证。这是全球首创,因为没有其他操作系统或可信执行环境 (TEE) 达到该安全级别。相比之下,移动安全市场上第二安全的 TEE(对于…

C语言菜鸟入门·数据结构·链表超详细解析

目录 1. 单链表 1.1 什么是单链表 1.1.1 不带头节点的单链表 1.1.2 带头结点的单链表 1.2 单链表的插入 1.2.1 按位序插入 (1)带头结点 (2)不带头结点 1.2.2 指定结点的后插操作 1.2.3 指定结点的前插操作 1.3 …

如何对人工智能系统进行测试|要点,方法及流程

当今社会,人工智能发展非常快。现在人工智能的发展已经渗透到了我们生活的方方面面,自动驾驶、或者我们手机里经常用到的一些应用都或多或少涉及到了一些人工智能的功能,比如说美图秀秀、新闻推荐、机器翻译以及个性化的购物推荐等等都涉及到…

视频监控汇聚平台LntonCVS视频监控管理平台解决方案和常见的接入方式

一、视频融合平台 LntonCVS是一款支持多种协议和设备接入的视频汇聚流媒体平台。它能够统一管理和整合不同品牌、不同协议的视频资源,构建视频数据资源池,并通过视频资源目录为各类业务场景提供丰富、实时、高清的视频资源。 二、接入方式 1. 前端设备…

视频汇聚平台EasyCVR接入移动执法记录仪,视频无法播放且报错500是什么原因?

GB28181国标视频汇聚平台EasyCVR视频管理系统以其强大的拓展性、灵活的部署方式、高性能的视频能力和智能化的分析能力,为各行各业的视频监控需求提供了优秀的解决方案。视频智能分析平台EasyCVR支持多协议接入,兼容多类型的设备,包括IPC、NV…

【unittest】TestSuite搭建测试用例示例二

1.1 打开串口示例 常用的模组则包含AT指令测试,或串口数据测试,则可添加串口配置,将指令通过串口发送出去,如下所示: import serial def open_serial_port(port, baudrate115200, timeout2): try: # 创建并配置串…

Cocos Creator2D游戏开发(10)-飞机大战(8)-计分和结束

现在游戏基本能完了, 飞机能发射子弹,打了敌机,敌机也能炸; 接下来要做计分了; 步骤: 搞出一个lable让lable显示炸了多少飞机 开搞: ①创建一个Lable标签 ② root.ts文件 添加 property(Label) player_score: Label; // 标签属性 标签绑定 ③ 代码添加 注册 然后回调 contac…

计算机网络-数据链路层

基本概念 数据链路和链路 链路:指的是从一个节点到相邻节点的一段物理线路,且中间没有任何其他的交换节点 数据链路:传输数据时,除了一条物理线路,还需要一些必要通信协议来控制这些传输。 数据链路层的三个基本问…

【架构】客户端优化

这篇文章总结一下服务器网关及之前部分的优化,如客户端的优化,CDN/DNS等。 这里我们先谈一谈客户端缓存优化的手段。一般我们后端在说到缓存,第一时间想到的往往是redis,其实缓存在架构层次还有很多其他可以实现的地方&#xff0…

度言软件介绍

度言软件管理员操作后台 https://www.duyansoft.com企业后台为公司管理员操作后台,共计有七个功能版块 控制台 成员管理——员工管理 成员管理——员工管理(添加员工) 成员管理——团队管理 公司管理员可以新建/编辑/删除团队&#xff0c…

SSM整合快速学习

目录 步骤: 一、环境搭建 1.创建JdbcConfig配置类 2.创建JdbcConfig配置类 3.创建MybatisConfig配置类 4.创建jdbc.properties 5.创建SpringMVC配置类 6.创建Web项目入口配置类 二、功能模块开发 步骤1:创建数据库及表 步骤2:编写模型类 步骤3:编写Dao接…

Java面试题--JVM大厂篇之Java中Parallel GC的调优技巧与最佳实践

目录 引言: 正文: 1. 理解Parallel GC的工作原理 2. 常见痛点与解决方案 痛点一:长时间暂停 痛点二:频繁的Minor GC 痛点三:内存溢出 3. 调优参数推荐 4. 实战经验分享 结束语: 引言:…