认识所有权

专栏简介:本专栏作为Rust语言的入门级的文章,目的是为了分享关于Rust语言的编程技巧和知识。对于Rust语言,虽然历史没有C++、和python历史悠远,但是它的优点可以说是非常的多,既继承了C++运行速度,还拥有了Java的内存管理,就我个人来说,还有一个优点就是集成化的编译工具cargo,语句风格和C++极其相似,所以说我本人还是比较喜欢这个语言,特此建立这个专栏,作为学习的记录分享。

日常分享:每天努力一点,不为别的,只是为了日后,能够多一些选择,选择舒心的日子,选择自己喜欢的人!


目录

Rust独有的所有权

所有权

所有权的规则

String类型

内存与分配

 变量数据的移动

字面值

String类型

变量数据的克隆与拷贝

克隆

栈上数据的拷贝

所有权与函数

返回值与作用域

引用与借用

可变引用

悬垂引用

引用的规则

Slice类型

字符串字面值是slice

 字符串slice作为参数

其他类型的slice

 总结


Rust独有的所有权

所有权这个特性时Rust独有的,前面第一章我们说了,Rust语言集合了C++,java的优点,而为了解决C++中垃圾无法回收,容易造成内存泄漏的特点,Rust中提出了所有权这个概念。所以说,认识所有权才是掌握Rust的必不可少的部分。

所有权

Rust的核心功能之一是所有权。下面我们来认识认识所有权。

学习过c++和java的人应该知道,在这两种语言中,c++需要开发者自己分配和释放内存,这种情况下很多开发者会在使用了内存而不释放,导致内存泄漏。而Java则是解决了这种问题,他采用的是垃圾回收机制,在程序运行的时候自动的寻找不再使用的内存。而Rust使用的则是所有权管理,简单的来说,就是在程序编译时就会进行规格检查,提前检测出可能会存在内存泄漏等问题。其实这个有点和微软公司下的visual studio编译器很相似,对内存管理很严格。

提到栈,很多人应该能想到数据结构中的栈,栈的特点就是“先进后出”,栈的内存是连续的,存放数据总是按照顺序方式存入,而在栈中,存入的数据必须是大小固定的,且已经知道的,在C++中,在开辟内存空间的时候,一般都是在栈上开辟的,所以必须要指明数据类型,以此来告诉系统开辟的空间是固定且已知大小的。

我们定义的指针则是在堆区,因为我们是不知道具体的内存大小的,只能分配一个足够大的内存空间。。 堆是缺乏组织的:当向堆放入数据时,你要请求一定大小的空间。内存分配器(memory allocator)在堆的某处找到一块足够大的空位,把它标记为已使用,并返回一个表示该位置地址的 针(pointer)。

所以说,当我们定义数据的内存大小不会变的时候就在栈区开辟空间,如果不确定会不会改变数据的大小,则在堆区开辟空间。

所有权的规则

  1. Rust 中的每一个值都有一个 所有者owner)。
  2. 值在任一时刻有且只有一个所有者。
  3. 当所有者(变量)离开作用域,这个值将被丢弃。

记住,和其他语言一样,变量(所有者)值只在对应的作用域起作用,超出作用域,则无法使用。

String类型

这里我们将String类型提出来单独讲解,其实是因为String类型有点独特,它是可变的,不再是固定的,所以说,将String类单独提出来讲解,而不是归于字面值。String类型的数据被分配在堆区,所以大小可以随时变化。

fn main()
{
  let mut s=String::from("Hello"); //从字面值中获取字符串
  println!("{}", s); 
  s.push_str(",World");  //在字符串s后追加字符串 
  println!("{}",s);
}

对于String类型中的一些函数, 大家可以去官网查看,这里就不过多介绍,至于String类型中函数的调用方法,会在后面将到。

内存与分配

前面说了,String类型是在堆区上分配内存的,最开始系统并不知道你需要多大的内存,就分配一个很大的内存空间,

  • 必须在运行时向内存分配器(memory allocator)请求内存。
  • 需要一个当我们处理完 String 时将内存返回给分配器的方法。

第一个部分是我们完成,在获取字面值的时候就请求分配内存,这在任何一门语言中均适用。

第二部分就需要根据不同语言的特性来进行操作了。比如说Java有垃圾回收机制,他会记录并清除不再使用的内存,我们不需要太多的关心内存。然而,像C++,Python这种没有垃圾回收机制的语言,需要自己去释放,比如说,C++中的析沟函数,会在类创建并使用完毕后自动释放内存,这是由于析沟函数自动调用了drop()函数。对于一些变量,我们也需要人为的去释放,使用drop()或者delete()函数。

但是在Rust中,当变量离开他所在的作用域的时候就会自动释放。Rust自动调用了特殊的函数,drop()函数,也可以自己手动调用。

fn main()
{
  let mut s=String::from("Hello"); //从字面值中获取字符串
  println!("{}", s); 
  s.push_str(",World");  //在字符串s后追加字符串 
  println!("{}",s);
  drop(s); //释放掉内存,所有者s不再存在
}

 变量数据的移动

字面值

fn main()
{
  let s1=2;
  let s2=s1;
  println!("{}", s1);
  println!("{}", s2);
}

如上述例子,先定义了一个变量s1,然后绑定到数值5上,再定义一个变量s2绑定到s1上,也就是说,两个变量的值都是5.由于他们在定义的时候就已经指明了数值大小,所以这两个变量存放在栈区,所以是按照顺序存放。两个变量都有效。那如果是存放在堆区的又该如何?

String类型

fn main()
{
  let s1=String::from("Hello!");
  let s2=s1;
  println!("{}", s1);
  println!("{}", s2);
}

上述例子,运行你会发现报错了,显示s1值不存在,这是为什么?我们先来介绍一下String类型的数据的概念,然后在来解释这个问题。

我们以s1为例:

Four tables: two tables representing the stack data for s1 and s2, and each points to its own copy of string data on the heap.

 

 

string类型的数据由三部分组成: 一个指向存放字符串内容内存的指针,一个长度,和一个容量。这一组数据存储在栈上。右侧则是堆上存放内容的内存部分。

长度表示 String 的内容当前使用了多少字节的内存。容量是 String 从分配器总共获取了多少字节的内存。

当我们将s2绑定到s1上的时候:

图示

也就是说,只拷贝了栈上的数据,而堆上的数据则没有被拷贝,两个变量共同指向堆区数值。 回到上面的问题,为什么s1会不存在,这是由于,如果s1存在,那么将有两个变量同时拥有同一个字面值,在离开作用域时,系统会自动调用drop()函数,这时就会出现两个变量都会被释放,就出现了二次释放double free)的错误。这是不被允许的。所以Rust在s2绑定到s1上时,就将s1清除。从而不能再使用。

 Compiling number v0.1.0 (/home/ddb/文档/number)
error[E0382]: borrow of moved value: `s1`
 --> src/main.rs:5:18
  |
3 |   let s1=String::from("Hello!");
  |       -- move occurs because `s1` has type `String`, which does not implement the `Copy` trait
4 |   let s2=s1;
  |          -- value moved here
5 |   println!("{}", s1);
  |                  ^^ value borrowed here after move
  |
  = note: this error originates in the macro `$crate::format_args_nl` which comes from the expansion of the macro `println` (in Nightly builds, run with -Z macro-backtrace for more info)

For more information about this error, try `rustc --explain E0382`.
error: could not compile `number` due to previous error

变量数据的克隆与拷贝

克隆

和其他语言一样,Rust也提供了克隆的方法。可以理解为深拷贝,将堆区的数据也拷贝一份。

fn main()
{
  let s1=String::from("Hello!");
  let s2=s1.clone();
  println!("{}", s1);
  println!("{}", s2);
}

栈上数据的拷贝

fn main()
{
  let s1=10;
  let s2=s1;
  println!("{}", s1);
  println!("{}", s2);
}

这里的一个例子,两个变量都是在栈上,所以他们的数据之间进行的是拷贝,两个变量都可以存在。可以实现copy的数据类型有:

  • 所有整数类型,比如 u32
  • 布尔类型,bool,它的值是 truefalse
  • 所有浮点数类型,比如 f64
  • 字符类型,char
  • 元组,当且仅当其包含的类型也都实现 Copy 的时候。比如,(i32, i32) 实现了 Copy,但 (i32, String) 就没有。

所有权与函数

将值传递给函数与给变量赋值的时候原理极为相似。所以,向函数传递值可能会移动或者复制。

fn main()
{
  let s=String::from("Hello, world!"); //s进入作用域
  String_move(s); //s的值移动到函数里,后面s不再有效
  let x=10;//x进入作用域
  number_copy(x); //x进入函数里,但是x是i32的,所以是复制进来,后面继续有效.
  println!("{}", x);
}
fn String_move(something: String) //something进入作用域
{
  println!("{}",something);
} //离开作用域,调用drop函数,something失效。
fn number_copy(number:i32) //number进入作用域,开始起作用
{
  println!("{}",number);
}//移出作用域

返回值与作用域

fn main()
{
  let s1=givs_ownership();
  let s2=String::from("test");
  let s3=takes_and_gives_back_ownership(s2); //s2被移动到函数中,返回值给s3,不再起作用.

} //s1,s2,s3均离开作用域,不起作用,但s2已被移走,不会发生什么。
fn givs_ownership()->String
{
  let something=String::from("something");  // "something"进入作用域
  return something;//返回"something",并移出给调用的函数
}
fn takes_and_gives_back_ownership(a_String:String)->String
{
  a_String; //返回a_String,并移出给调用函数。
}

上面两个例子都是在说明所有权问题,一个值只能有一个所有者,所以,上面不同的函数,不同的数据类型之间使用的是不同的方法,有的是移动,有的是克隆。在这一点上,一定要注意区分。

注意,函数在返回值上面,可以返回多个值,但是是以元组的方式返回。

fn main()
{
  let s1=String::from("hello world");
  let (num,s2)=string_length(s1);
  println!("{} {}",num,s2);

}
fn string_length(s:String)->(usize,String) 
{
  let length=s.len();
  return (length,s);
}

引用与借用

引用reference)像一个指针,因为它是一个地址,我们可以由此访问储存于该地址的属于其他变量的数据。 与指针不同,引用确保指向某个特定类型的有效值。

fn main() 
{
  let s=String::from("hello");
  let length:usize =string_length(&s);
  println!("the length of the string s is: {}",length);

}
fn string_length(s: &String) -> usize
{
  return s.len();

}

 上面函数中“&s”表示的是建立一个引用,指向s的数据值,但是并不拥有它,也就不会有所有权这个东西。所以在离开作用域后对原本的数据值没什么影响。

在Rust中,我们把创建一个引用的行为称为借用

可变引用

fn main() 
{
  let mut s=String::from("hello");
  let length:String =string_length(&mut s);
  println!("{}",s);
  println!("position is  {}",length);

}
fn string_length(s: &mut String) ->String
{
  s.push_str(",world!");
  let m:String = String::from("Successfull!");
  return m;
}

上面的代码就是一个可变引用的实现,我们可以看到,可变引用就是在引用前加一个mut关键字。

上面我们说过,引用只是暂时借用数据,并不拥有所有权,所以,一个变量被创建了可变引用的时候,他只能被创建一次,否则会报错,这是由于,当你创建了多个可变引用的时候,他们都可以更改原本的数据,这个时候系统就不会知道那一个改变在前面,那一个在后面。就会出现混乱。

例如:

ddb@ddb-NBLK-WAX9X:~/文档/number$ cargo run
   Compiling number v0.1.0 (/home/ddb/文档/number)
error[E0499]: cannot borrow `sln` as mutable more than once at a time
 --> src/main.rs:5:10
  |
4 |   let m1=&mut sln;
  |          -------- first mutable borrow occurs here
5 |   let m2=&mut sln;
  |          ^^^^^^^^ second mutable borrow occurs here
6 |   println!("{},{}",m1,m2);
  |                    -- first borrow later used here

For more information about this error, try `rustc --explain E0499`.
error: could not compile `number` due to previous error

 出现上面这种情况,在官方的说法中是数据竞争。导致这种情况的原因:

  • 两个或更多指针同时访问同一数据。
  • 至少有一个指针被用来写入数据。
  • 没有同步数据访问的机制。

 除了不能同时拥有多个可变引用,还不能同时存在可变与不可变引用。举个简单的例子,你有一个玩具,有人来借,然后会原样还回,但是有人却改变了他的模样,你还回去的玩具和他的不一样,是不是就会出现矛盾。

 let mut sln=String::from("I like Rust!");
  let m1=&sln;
  let m2=&sln;
  let m3=&mut sln;
  println!("{},{},{}",m1,m2,m3);

像上面这种情况就会出现报错。

悬垂引用

在具有指针的语言中,很容易通过释放内存时保留指向它的指针而错误地生成一个 悬垂指针dangling pointer),所谓悬垂指针是其指向的内存可能已经被分配给其它持有者。相比之下,在 Rust 中编译器确保引用永远也不会变成悬垂状态:当你拥有一些数据的引用,编译器确保数据不会在其引用之前离开作用域。

fn main() {
    let reference_to_nothing = dangle();
}

fn dangle() -> &String {
    let s = String::from("hello");

    &s
}

这里出现了报错,这是由于s在离开作用域后就失去了作用,不再有任何的作用,所以引用肯定也没作用了。

引用的规则

  • 在任意给定时间,要么 只能有一个可变引用,要么 只能有多个不可变引用。
  • 引用必须总是有效的。

Slice类型

slice 允许你引用集合中一段连续的元素序列,而不用引用整个集合。slice 是一类引用,所以它没有所有权。


fn main()
{
  let mut s=String::from("hello world");
  let world=first_world(&s);
  println!("{}", world);
  s.clear();  //清空字符串
}
fn first_world(s: &String) -> usize {
  let bytes = s.as_bytes();

  for (i, &item) in bytes.iter().enumerate() {
      if item == b' ' {
          return i;
      }
  }

  s.len()
}

看上面,上面的这段代码表示的是找到空格的所在位置,对于这个函数,使用了很多的库函数才求出索引值,那么有没有简单方法?

这里我们必须提到字符串slice,在Python中,我们可以直接使用索引值求的字符串中的部分字符串,而在Rust中,也有这种机制,


fn main() {
    let s = String::from("hello world");

    let hello = &s[0..5];
    let world = &s[6..11];
    println!("{},{}", world, hello);
}

和Python一样,他的索引方式也有很多,比如s[..3],s[..],s[2..]他们分别表示的是从0下标开始到3位置,从0开始到尾部结束,从2开始到结束。

字符串字面值是slice

前面我们说过,字符串字面值和String的区别,字符串字面值是不可变的,这是由于字符串字面值的数据类型就是&str。

例如:

let s = "Hello, world!";

这里 s 的类型是 &str:它是一个指向二进制程序特定位置的 slice。这也就是为什么字符串字面值是不可变的;&str 是一个不可变引用。

 字符串slice作为参数

slice也可以作为函数的返回数据类型和参数类型:


fn main() {
  let s=String::from("Hello,world");
  let mun=&s[..];
  let a=Slice_from(mun);
  println!("{}",a);
}
fn Slice_from(s:&str)->&str {
  let sl:String = String::from("Hello, world");
  let world = &sl[6..11];
  let hello=&s[0..5];
  println!("{}", world);
  return hello;
}

其他类型的slice

字符串 slice,正如你想象的那样,是针对字符串的。不过也有更通用的 slice 类型。考虑一下这个数组:

let a = [1, 2, 3, 4, 5]; 

就跟我们想要获取字符串的一部分那样,我们也会想要引用数组的一部分。我们可以这样做:

let a = [1, 2, 3, 4, 5]; 
let slice = &a[1..3];
 assert_eq!(slice, &[2, 3]); 

 总结

本节内容比较多,主体意思就是说在Rust中,Rust 语言提供了跟其他系统编程语言相同的方式来控制你使用的内存,但拥有数据所有者在离开作用域后自动清除其数据的功能意味着你无须额外编写和调试相关的控制代码。所以说,只要知道Rust语言的特有机制,学起来就会简单很多。

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

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

相关文章

zookeeper的部署

一 先下载zookeeper 二 解压包 三 修改配置文件 四 把配好文件传到其他的节点上面 五 在每个节点的dataDir指定的目录下创建一个 myid 的文件 六 配置zook的启动脚本 七 设置开机自启 八 分别启动 九查看当前状态service zookeeper status 十 总结 一 先下载zookeeper …

Vue常见的事件修饰符

1.prevent:阻止默认事件(常用) 2. stop:阻止事件冒泡(常用) 3. once:事件只触发一次(常用) 4.captrue:使用事件的捕捉模式(不常用) 5.self:只有event.target是当前操作的元素时才触发事件(不常用) 6.passive:事件的默认行为立即执行,无需等待事件回调执行完毕(不常用…

网关 GateWay 的使用详解、路由、过滤器、跨域配置

一、网关的基本概念 SpringCloudGateway网关是所有微服务的统一入口。 1.1 它的主要作用是: 反向代理(请求的转发) 路由和负载均衡 身份认证和权限控制 对请求限流 1.2 相比于Zuul的优势: SpringCloudGateway基于Spring5中…

Go微服务实践 - Rpc核心概念理解

概述 从0研究一下Golang已经Golang的微服务生态体系,Golang的微服务首先要从Rpc开始,在升级到Grpc,详细介绍这些技术点都在解决什么技术问题。 Rpc Rpc (Remote Procedure Call) 远程过程调用,简单的理解是一个节点请求另一个节…

集成学习算法是什么?如何理解集成学习?

什么是集成学习? 集成学习通过建立几个模型来解决单一预测问题。它的工作原理是生成多个分类器/模型,各自独立地学习和作出预测。这些预测最后结合成组合预测,因此优于任何一个单分类的做出预测。 机器学习的两个核心任务 任务一&#xff1…

【JavaEE】Spring Boot - 日志文件

【JavaEE】Spring Boot 开发要点总结(3) 文章目录 【JavaEE】Spring Boot 开发要点总结(3)1. 日志有什么作用2. 日志格式2.1 日志框架原理 3. 日志的打印3.1 System.out.println3.2 使用日志框架3.3 日志级别3.3.1 设置默认日志显…

自监督去噪:Noise2Self原理分析及实现 (Pytorch)

文章地址:https://arxiv.org/abs/1901.11365 代码地址: https://github.com/czbiohub-sf/noise2self 要点   Noise2Self方法不需要信号先验信息、噪声估计信息和干净的训练数据。唯一的假设就是噪声在测量的不同维度上表现出的统计独立性,而真实信号表现出一定的…

Vue Router 的query和params的区别?

区别一: (1)query相当于get请求,页面跳转的时候可以在地址栏看到请求参数 (2)params相当于post请求,参数不会在地址栏中显示,所以用params传值相对安全 (简记&#xff1…

C++ | C++11新特性(上)

目录 前言 一、列表初始化 二、声明 1、auto 2、decltype 3、nullptr 三、STL容器的变化 四、右值引用与移动语义 1、左值与左值引用 2、右值与右值引用 3、右值引用与左值引用的比较 4、右值引用的场景及意义 (1)做参数 (2&a…

Python连接Hive实例教程

一 Python连接hive环境实例 经在网络查询相关的教程,发现有好多的例子,发现连接底层用的的驱动基本都是pyhive和pyhs2两种第三方库的来连接的 hive,下面将简介windows 10 python 3.10 连接hive的驱动程序方式,开发工具:pycharm …

layui之layer弹出层的icon数字及效果展示

layer的icon样式 icon如果在信息提示弹出层值(type为0)可以传入0-6,icon与图标对应关系如下: 如果是加载层(type为3)可以传入0-2,icon与图标对应关系如下:

基于fpga的电子时钟

文章目录 前言实验手册一、实验目的二、实验原理1.理论原理2.硬件原理 三、系统架构设计四、模块说明1.模块端口信号列表按键消抖模块(key)计数器模块(counter)蜂鸣器乐谱模块(music)蜂鸣器发声…

有没有好用的在线画图工具推荐?

绘画是设计师最常见的工作之一,设计师对在线绘画工具的要求越来越高,市场上也出现了各种在线绘画工具,让设计师不知道如何选择高质量的在线绘画工具,一个好的在线绘画工具不仅可以让你轻松绘画,而且可以让你的工作效率…

可视化高级绘图技巧100篇-总论

前言 优秀的数据可视化作品可以用三个关键词概括:准确、清晰、优雅。 准确:精准地反馈数据的特征信息(既不遗漏也不冗余,不造成读者疏漏&误读细节) 清晰:获取图表特征信息的时间越短越好 优雅&…

【ARM64 常见汇编指令学习 13 -- ARM 汇编 ORG 伪指令学习】

文章目录 ARM ORG 指令介绍UEFI 中对 ORG 指令的使用 ARM ORG 指令介绍 在ARM汇编中,"org"是一个汇编器伪指令,用于设置下一条指令的装入地址。"org"后面跟着的是一个表达式,这个表达式的值就是下一条指令的装入地址。如…

21、p6spy输出执行SQL日志

文章目录 1、背景2、简介3、接入3.1、 引入依赖3.2、修改database参数:3.3、 创建P6SpyLogger类,自定义日志格式3.4、添加spy.properties3.5、 输出样例 4、补充4.1、参数说明 1、背景 在开发的过程中,总希望方法执行完了可以看到完整是sql语…

侯捷 C++面向对象编程笔记——9 复合 委托

9 复合 委托 9.1 Composition 复合 类似于c中结构里有结构——class里有class deque 是一个已经存在的功能很多的类(两头进出的队列);利用deque的功能来实现queue的多种操作 该例只是复合的一种情况——设计模式 Adapter 9.1.1 复合下的构造…

C# 完成串口通信RS485

C# 完成串口通信RS485|RS232上下位机交互 第零步: 我用的是电脑usb 转串口的所以首先是驱动程序下载,我们用的是CH341 下载地址:https://www.wch.cn/downloads/CH341SER_EXE.html 第一步:连接机器 RS485 上面有三个端子&#xf…

【Python】Locust持续优化:InfluxDB与Grafana实现数据持久化与可视化分析

目录 前言 influxDB 安装运行InfluxDB 用Python 上报数据到influxdb ocust 数据写入到 influx Locust的生命周期 上报数据 优化升级 配置Grafana 总结 资料获取方法 前言 在进行性能测试时,我们需要对测试结果进行监控和分析,以便于及时发现问…

flutter-GridView使用

先看效果 代码实现 import package:app/common/util/k_log_util.dart; import package:app/gen/assets.gen.dart; import package:app/pages/widget/top_appbar.dart; import package:flutter/cupertino.dart; import package:flutter/material.dart; import package:flutter_…