欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

rust智能指針的具體使用

 更新時間:2023年12月07日 15:00:27   作者:int8  
智能指針是一些數(shù)據(jù)結(jié)構(gòu),它們的行為類似于指針但擁有額外的元數(shù)據(jù)和附加功能,本文就來介紹一下rust智能指針的具體使用,感興趣的可以了解一下

一、智能指針是什么

指針是一個存儲內(nèi)存地址的變量。這個地址指向一些其他數(shù)據(jù)。
智能指針是一類數(shù)據(jù)結(jié)構(gòu),它們類似指針,但是擁有額外的功能。智能指針的概念起源于C++。Rust標準庫提供了許多智能指針,比如String和Vec<T>,雖然我們并不這么稱呼它們,但這些類型都屬于智能指針。
智能指針通常使用結(jié)構(gòu)體實現(xiàn)。智能指針與常規(guī)結(jié)構(gòu)體的區(qū)別在于智能指針實現(xiàn)了Deref和Drop trait。Deref trait使智能指針表現(xiàn)的像引用一樣,這樣就可以編寫既用于引用、又用于智能指針的代碼。Drop trait允許我們自定義智能指針離開作用域時的行為。

在Rust中,引用和智能指針的一個區(qū)別是引用是一類只借用數(shù)據(jù)的指針;智能指針則擁有數(shù)據(jù)的所有權(quán)。

二、最常用的一些智能指針

1.Box<T>,用于在堆上分配
2.Rc<T>,一個引用計數(shù)類型,其數(shù)據(jù)可以有多個所有者
3.Ref<T> 和RefMut<T>,通過RefCell<T> 訪問(RefCell<T>是一個在運行時而不是在編譯時執(zhí)行借用規(guī)則的類型)

(一)Box指針

Box<T>類型是一個智能指針,因為它實現(xiàn)了Deref trait和Drop trait。
box把值放在堆上而不是棧上。留在棧上的則是指向堆數(shù)據(jù)的指針。除了數(shù)據(jù)被儲存在堆上而不是棧上之外,box沒有性能損失。不過也沒有很多額外的功能。
1.創(chuàng)建Box
使用new函數(shù)創(chuàng)建
例子

fn main() {
     let var_i32 = 5; // 默認數(shù)據(jù)保存在 棧 上
     let b = Box::new(var_i32); // 使用Box后數(shù)據(jù)會存儲在堆上
     println!("b = {}", b);
}

b離開作用域時,它將自動釋放。這個釋放包括b本身(位于棧上)和它所指向的數(shù)據(jù)(位于堆上)。

2.使用box
像使用引用一樣使用box。
使用解引用操作符 * 解引用box

fn main() {
     let x = 5; // 值類型數(shù)據(jù)
     let y = Box::new(x); // y是一個智能指針,指向堆上存儲的數(shù)據(jù)5
     println!("{}",5==x);
     println!("{}",5==*y); // 為了訪問y存儲的具體數(shù)據(jù),需要解引用
}
編譯運行結(jié)果如下
true
true
直接使用  5 == y  會返回false

3.使用Box創(chuàng)建遞歸類型
Rust需要在編譯時知道類型占用多少空間。一種無法在編譯時知道大小的類型是遞歸類型,其值的一部分可以是自身類型的另一個值。這種嵌套可以是無限的,所以Rust不知道遞歸類型需要多少空間。不過box有一個已知的大小,所以通過在遞歸類型定義中插入box,就可以創(chuàng)建遞歸類型了。一個常見遞歸類型就是鏈表。
實例

enum List {
     Cons(i32, List),
     Nil,
}
use crate::List::{Cons, Nil};
fn main() {
     let list = Cons(1, Cons(2, Cons(3, Nil)));//使用這個list來儲存1, 2, 3
}

第一個Cons儲存1和另一個List值。這個List是一個Cons值,此cons儲存2和下一個List值。這個list又是一個cons,儲存3和值為Nil的List。這段代碼編譯錯誤。因為這個類型 “有無限的大小”。
因為Box<T> 是一個指針,它的大小是確定的,所以將Box作為Cons的成員,這樣List的大小就確定了。

enum List {
     Cons(i32, Box<List>),
     Nil,
}
use crate::List::{Cons, Nil};
fn main() {
     let list = Cons(1,
         Box::new(Cons(2,
              Box::new(Cons(3,
                  Box::new(Nil))))));
}

三、兩個特性

(一)Deref Trait

1.Deref是由Rust標準庫提供的一個特性。
實現(xiàn)Deref之后就能把智能指針當作引用使用,相當于重載解引用運算符*。
Deref中包含deref()方法。
deref()方法用于引用self實例并返回一個指向內(nèi)部數(shù)據(jù)的指針。

例子

use std::ops::Deref;
struct DerefExample<T> {
     value: T
}
impl<T> Deref for DerefExample<T> {
     type Target = T;
     fn deref(&self) -> &Self::Target {
         &self.value
     }
}
let x = DerefExample { value: 'a' };
assert_eq!('a', *x);

范例

use std::ops::Deref;
struct MyBox<T>(T);
impl<T> MyBox<T> {
     fn new(x:T)-> MyBox<T> {
         MyBox(x)
     }
}
impl<T> Deref for MyBox<T> {
     type Target = T;
     fn deref(&self) -> &T {
         &self.0
     }
}
fn main() {
     let x = 5;
     let y = MyBox::new(x); // 調(diào)用new() 返回創(chuàng)建一個結(jié)構(gòu)體實例
     println!("5==x is {}",5==x);
     println!("5==*y is {}",5==*y); // 解引用y
     println!("x==*y is {}",x==*y); // 解引用y
}
編譯運行結(jié)果如下
5==x is true
5==*y is true
x==*y is true

每次使用 * 時, * 運算符都被替換成先調(diào)用deref方法再使用 * 解引用的操作,且只會發(fā)生一次,不會無限遞歸替換 * 操作符,解引用出i32類型的值就停止了

2.DerefMut trait用于重載可變引用的 * 運算符

3.Deref隱式轉(zhuǎn)換
Deref隱式轉(zhuǎn)換將實現(xiàn)了Deref的類型的引用轉(zhuǎn)換為另一種類型的引用。例如,將&String轉(zhuǎn)換為&str,因為String實現(xiàn)了Deref因此可以返回&str。Deref強制轉(zhuǎn)換是Rust在函數(shù)或方法傳參上的一種便利操作,并且只能作用于實現(xiàn)了Deref的類型。當這種特定類型的引用作為實參傳遞給和形參類型不同的函數(shù)時將自動轉(zhuǎn)換類型。這時會有一系列的deref方法被調(diào)用,把我們提供的類型轉(zhuǎn)換成了形參所需的類型。
Deref隱式轉(zhuǎn)換使Rust程序員在調(diào)用函數(shù)時無需使用過多& 和 *。這個功能方便我們編寫同時作用于引用或智能指針的代碼。

實例

//還是上面的MyBox<T>
fn hello(name: &str) {
     println!("Hello, {name}!");
}
let m = MyBox::new(String::from("Rust"));
hello(&m);

因為Deref隱式轉(zhuǎn)換,使用MyBox<String>的引用作為參數(shù)是可行的。
因為MyBox<T>實現(xiàn)了Deref,Rust可以通過deref將&MyBox<String>變?yōu)?amp;String。而String也實現(xiàn)了Deref,Rust再次調(diào)用deref將&String變?yōu)?amp;str,這就符合hello函數(shù)的定義了。

如果沒有Deref強制轉(zhuǎn)換,要把&MyBox<String>類型的值傳給hello函數(shù),則不得不編寫如下代碼

let m = MyBox::new(String::from("Rust"));
hello(&(*m)[..]);

(*m)MyBox<String>解引用為String,接著&和[..]將String轉(zhuǎn)換成&str。
沒有Deref強制轉(zhuǎn)換的話,所有這些符號混在一起將難以讀寫和理解。Deref強制轉(zhuǎn)換會自動執(zhí)行這些轉(zhuǎn)換。這些轉(zhuǎn)換發(fā)生在編譯時,所以沒有運行時損耗!

Deref隱式轉(zhuǎn)換有三種情形:
(1)當T實現(xiàn)Deref Target=U 時從 &T到 &U。
(2)當T實現(xiàn)DerefMut Target=U 時從 &mut T到 &mut U。
(3)當T實現(xiàn)Deref Target=U 時從 &mut T到 &U。
第一種情況表明如果有一個 &T,而T實現(xiàn)了返回U類型的Deref,則可以直接得到 &U。
第二種情況表明可變引用也有著相同的行為。
第三個情況將可變引用強轉(zhuǎn)為不可變引用。但是反過來是不行的,不可變引用永遠也不能強轉(zhuǎn)為可變引用。
這三種情況下,T類型都自動實現(xiàn)了U類型的所有方法。

(二)Drop Trait

Rust中的析構(gòu)函數(shù)是由Drop trait提供的drop()方法。
Drop Trait只有一個方法drop() 。
實現(xiàn)了Drop特質(zhì)的結(jié)構(gòu)體在離開了它的作用域時會調(diào)用drop()方法。
例子

use std::ops::Deref;
struct MyBox<T>(T);
impl<T> MyBox<T> {
     fn new(x:T)->MyBox<T>{
         MyBox(x)
     }
}
impl<T> Deref for MyBox<T> {
     type Target = T;
     fn deref(&self) -< &T {
         &self.0
     }
}
impl<T> Drop for MyBox<T>{
     fn drop(&mut self){
         println!("dropping MyBox object from memory ");
     }
}
fn main() {
     let x = 50;
     MyBox::new(x);
     MyBox::new("Hello");
}
編譯運行結(jié)果如下
dropping MyBox object from memory
dropping MyBox object from memory

到此這篇關(guān)于rust智能指針的具體使用的文章就介紹到這了,更多相關(guān)rust智能指針內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 利用Rust實現(xiàn)一個簡單的Ping應(yīng)用

    利用Rust實現(xiàn)一個簡單的Ping應(yīng)用

    這兩年Rust火的一塌糊涂,甚至都燒到了前端,再不學(xué)習(xí)怕是要落伍了。最近翻了翻文檔,寫了個簡單的Ping應(yīng)用練練手,感興趣的小伙伴可以了解一下
    2022-12-12
  • rust中async/await的使用示例詳解

    rust中async/await的使用示例詳解

    在Rust中,async/await用于編寫異步代碼,使得異步操作更易于理解和編寫,通過使用await,在async函數(shù)或代碼塊中等待Future完成,而不會阻塞線程,允許同時執(zhí)行其他Future,這種機制簡化了異步編程的復(fù)雜性,使代碼更加直觀
    2024-10-10
  • Rust 的 into_owned() 方法實例詳解

    Rust 的 into_owned() 方法實例詳解

    into_owned是Rust語言中std::borrow::Cow 枚舉的一個方法,into_owned確保了調(diào)用者獲得數(shù)據(jù)的獨立所有權(quán),無論Cow之前是引用還是已經(jīng)擁有數(shù)據(jù),本文給大家介紹Rust 的 into_owned() 方法,感興趣的的朋友跟隨小編一起看看吧
    2024-03-03
  • rust流程控制的具體使用

    rust流程控制的具體使用

    在Rust中,控制流包括條件語句、循環(huán)和匹配模式等,用于實現(xiàn)程序的邏輯和流程控制,本文就來詳細的介紹一下,感興趣的可以了解一下
    2023-12-12
  • 深入理解Rust中Cargo的使用

    深入理解Rust中Cargo的使用

    本文主要介紹了深入理解Rust中Cargo的使用,Cargo簡化了項目的構(gòu)建過程,提供了依賴項管理,以及一系列方便的工作流程工具,下面就來具體的介紹一下如何使用,感興趣的可以了解一下
    2024-04-04
  • rust開發(fā)環(huán)境配置詳細教程

    rust開發(fā)環(huán)境配置詳細教程

    rust是一門比較新的編程語言,2015年5月15日,Rust編程語言核心團隊正式宣布發(fā)布Rust 1.0版本,這篇文章主要介紹了rust開發(fā)環(huán)境配置 ,需要的朋友可以參考下
    2022-12-12
  • 詳解Rust中#[derive]屬性怎么使用

    詳解Rust中#[derive]屬性怎么使用

    在 Rust 中,#[derive] 是一個屬性,用于自動為類型生成常見的實現(xiàn),下面就跟隨小編一起來學(xué)習(xí)一下Rust中derive屬性的具體使用吧
    2024-11-11
  • rust生命周期詳解

    rust生命周期詳解

    生命周期是rust中用來規(guī)定引用的有效作用域,在大多數(shù)時候,無需手動聲明,因為編譯器能夠自動推導(dǎo),這篇文章主要介紹了rust生命周期相關(guān)知識,需要的朋友可以參考下
    2023-03-03
  • Rust 數(shù)據(jù)類型詳解

    Rust 數(shù)據(jù)類型詳解

    本文介紹了Rust編程語言中的標量類型和復(fù)合類型,標量類型包括整數(shù)、浮點數(shù)、布爾和字符,而復(fù)合類型則包括元組和數(shù)組,標量類型用于表示單個值,具有不同的表示和范圍,本文介紹的非常詳細,感興趣的朋友一起看看吧
    2025-01-01
  • Rust使用Channel實現(xiàn)跨線程傳遞數(shù)據(jù)

    Rust使用Channel實現(xiàn)跨線程傳遞數(shù)據(jù)

    消息傳遞是一種很流行且能保證安全并發(fā)的技術(shù),Rust也提供了一種基于消息傳遞的并發(fā)方式,在rust里使用標準庫提供的Channel來實現(xiàn),下面我們就來學(xué)習(xí)一下如何使用Channel實現(xiàn)跨線程傳遞數(shù)據(jù)吧
    2023-12-12

最新評論