热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

引用与借用:深入解析与实践应用

引用与借用是编程中重要的概念,涉及数据类型的管理和访问方式。引用允许在不获取所有权的情况下使用值,而解引用则通过解引用运算符`*`来访问引用所指向的实际数据。本文深入解析了引用与借用的机制,并通过具体的代码示例展示了其在实际编程中的应用,帮助读者更好地理解和掌握这些关键概念。

引用与借用



  • 引用是一种数据类型



  • & 符号就是引用



    • 允许使用值但不获取其所有权



  • 解引用dereferencing



    • 解引用运算符 *



fn main() {
let s1 = String::from("hello");
let len = calculate_length(&s1); // &s1 创建了一个指向 s1 的引用,但不拥有s1

println!("The length of '{}' is {}.", s1, len);
}
fn calculate_length(s: &String) -> usize { //参数为&String,不是String
s.len()
} // s 离开了作用域。但因为它并不拥有引用值的所有权,所以什么也不会发生


  • s 为 s1 的引用,s 指向 s1 (指针?




借用(borrowing



  • 把引用作为函数参数的行为叫做借用



  • 默认不允许修改引用的值

    fn main() {
    let s = String::from("hello");
    change(&s);
    }
    fn change(some_string: &String) {
    some_string.push_str(", world");
    }

    该代码不能编译




可变引用



  • &mut string就为一个可变引用类型

    fn main() {
    let mut s = String::from("hello");
    change(&mut s);
    }
    fn change(some_string: &mut String) {
    some_string.push_str(", world");
    }

    该代码可以正常编译



  • 限制:在同一时间,对于某一特定的数据,只能有一个可变引用



    • 尝试创建两个可变引用的代码将会失败

    let mut s = String::from("hello");
    let r1 = &mut s;
    let r2 = &mut s;

    println!("{}, {}", r1, r2);

    该代码不能编译



  • 这个限制的好处是 Rust 可以在编译时就避免数据竞争(出现数据竞争的情况时,rust在编译时就会报错



    • 数据竞争data race)类似于竞态条件,它可由这三个行为造成:

      • 两个或更多指针同时访问同一数据

      • 至少有一个指针被用来写入数据

      • 没有同步数据访问的机制





  • 可以创建新的作用域(通过大括号),以允许非同时拥有多个可变引用

    let mut s = String::from("hello");
    {
    let r1 = &mut s;
    } // r1 在这里离开了作用域,所以可以创建一个新的引用 r2

    let r2 = &mut s;


  • 另一个限制:不可以同时有一个可变引用与一个不可变引用



    • 但可以有多个不可变引用

    let mut s = String::from("hello");
    let r1 = &s; // 不可变引用
    let r2 = &s; // 不可变引用
    let r3 = &mut s; // 可变引用,不可以

    println!("{}, {}, and {}", r1, r2, r3);

    该代码不可以编译




悬垂引用(Dangling References



  • 悬垂指针dangling pointer),即一个指针引用了内存的某个地址,但这块内存可能已经事发并分配给其他人使用了



  • 在 Rust 中编译器可以确保引用永远也不会变成悬垂状态



    • 当你拥有一些数据的引用,编译器确保数据不会在其引用之前离开作用域。

    fn main() {
    let reference_to_nothing = dangle();
    }
    fn dangle() -> &String {
    let s = String::from("hello"); // s 在离开函数(离开作用域)后,就失效了
    &s //但对 s 的引用 &s 作为返回值返回了
    //即对 s 的引用 &s 就指向了一个被释放的内存地址
    }

    该代码不可以编译




引用的规则(总结



  • 在任意给定时间,要么 只能有一个可变引用,要么 只能有多个不可变引用

  • 引用必须一直是有效的



推荐阅读
  • 本题探讨了一种字符串变换方法,旨在判断两个给定的字符串是否可以通过特定的字母替换和位置交换操作相互转换。核心在于找到这些变换中的不变量,从而确定转换的可能性。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 探讨如何通过编程技术实现100个并发连接,解决线程创建顺序问题,并提供高效的并发测试方案。 ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • IneedtofocusTextCellsonebyoneviaabuttonclick.ItriedlistView.ScrollTo.我需要通过点击按钮逐个关注Tex ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
author-avatar
小洲相册居士
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有