網站首頁 編程語言 正文
前言
??Rust 生命周期機制是與所有權機制同等重要的資源管理機制,之所以引入這個概念主要是應對復雜類型系統中資源管理的問題。引用是對待復雜類型時必不可少的機制,畢竟在Rust 中復雜類型的數據不能被處理器輕易地復制和計算。但是為什么還有引入生命周期的概念呢,這是因為引用常常會導致非常復雜的資源管理問題。
1、所有權中的垂懸引用解析
先來看一下垂懸引用中所有權的變化:
{ let ans; { let x = 5; ans = &x; } println!("ans: {}", ans); }
這段代碼是不會通過 Rust編譯器的,原因是
ans
所引用的值已經在使用之前被釋放,borrowed value does not live long enough
意為x
有效時間太短。
紅色區域為
ans
的生命周期,綠色區域為x
的生命周期,很顯然綠色區域比紅色區域小得多,引用必須在值的生命周期以內才有效。
2、結構體中使用String 而不用&str 的原因
用一個函數 longer
解釋:
fn longer(s1: &str, s2: &str) -> &str { if s2.len() > s1.len() { s2 } else { s1 } } fn main() { let r; { let s1 = "rust"; let s2 = "ecmascript"; r = longer(s1, s2); } println!("{} is longer", r); }
longer 函數取
s1
和s2
兩個字符串切片中較長的一個返回其引用值。
這段代碼不會通過編譯,原因是返回值引用可能會返回過期的引用:
這段程序中雖然經過了比較,但 r 被使用的時候源值 s1 和 s2 都已經失效了。當然我們可以把 r 的使用移到 s1 和 s2 的生命周期范圍以內防止這種錯誤的發生。
對于函數來說,它并不能知道自己以外的地方是什么情況,它為了保障自己傳遞出去的值是正常的,必須遵循所有權原則消除一切危險,所以 longer 函數并不能通過編譯。
3、生命周期注釋
生命周期注釋是描述引用生命周期的辦法,雖然這樣并不能夠改變引用的生命周期,但可以在合適的地方聲明兩個引用的生命周期一致。
生命周期注釋用單引號開頭,跟著一個小寫字母單詞:
&i32 // 常規的引用 &'a i32 // 含有生命周期注釋的引用 &'a mut i32 // 可變型含有生命周期注釋的引用
讓我們用生命周期注釋改造 longer 函數:
fn longer<'a>(s1: &'a str, s2: &'a str) -> &'a str { if s2.len() > s1.len() { s2 } else { s1 } }
我們需要用泛型聲明來規范生命周期的名稱,函數返回值的生命周期將與兩個參數的生命周期一致
fn main() { let r; { let s1 = "rust"; let s2 = "ecmascript"; r = longer(s1, s2); println!("{} is longer", r); } } 運行結果:ecmascript is longer
注意:
Rust 自動推導類型的能力很強,如果上面的s1、s2不是字符串切片類型,而是字符串類型的話
r
得到的值會在{}
執行完后通過drop
自動清理掉。
4、結構體中使用字符串切片引用
之前的文章中說過結構體中也是可以使用字符串切片的,那么了解過生命周期的知識后就可以具體設計一個示例了:
fn main() { struct Str<'a> { content: &'a str } let s = Str { content: "string_slice" }; println!("s.content = {}", s.content); } //運行結果:s.content = string_slice
如果對結構體 Str 有方法定義:
impl<'a> Str<'a> { fn get_content(&self) -> &str { self.content } }
這里返回值并沒有生命周期注釋,早期 Rust 不支持生命周期自動判斷,所有的生命周期必須嚴格聲明,但主流穩定版本的 Rust 已經支持了這個功能,因此可以不加注釋。
5、靜態生命周期
生命周期注釋有一個特別的:'static
。
所有用雙引號包括的字符串常量所代表的精確數據類型都是 &'static str
。'static
所表示的生命周期從程序運行開始到程序運行結束,就相當于其他語言中的靜態全局變量。
6、泛型、特性與生命周期綜合使用
函數如下:
use std::fmt::Display; fn longest_with_an_announcement<'a, T>(x: &'a str, y: &'a str, ann: T) -> &'a str where T: Display { println!("Announcement! {}", ann); if x.len() > y.len() { x } else { y } }
這段程序出自 《Rust 圣經》,是一個同時使用了泛型、特性、生命周期 機制的程序,大家可以體驗一下 Rust
這種巧妙的組合,先有個體驗,到后面的學習中肯定會用到。
原文鏈接:https://blog.csdn.net/m0_58618795/article/details/127356414
相關推薦
- 2022-02-26 SpringSecurity 自定義JwtAuthorFilter基于JWT的Token驗證
- 2022-05-22 C#多線程編程Task用法詳解_C#教程
- 2022-07-21 linux環境下執行jar與終止
- 2021-12-05 Android?NDK開發(C語言基本數據類型)_Android
- 2022-01-22 oracle之連接查詢和子查詢
- 2023-07-30 el-table自定義合并行或列
- 2022-04-08 Unity?UGUI?按鈕綁定事件的?4?種方式匯總_C#教程
- 2022-04-09 IDEA中命令行安裝git報錯:error: failed to push some refs to
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支