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

Rust語言中級教程之指針

 更新時間:2023年05月06日 16:09:37   作者:QIAOPENGJUN  
Rust中共有三種類型的指針,分別為引用,解引用,智能指針,這篇文章主要介紹了Rust語言中級教程之指針,需要的朋友可以參考下

Rust語言中級教程

一、指針

什么是指針

  • 指針是計算機引用無法立即直接訪問的數(shù)據(jù)的一種方式(類比 書的目錄)
  • 數(shù)據(jù)在物理內(nèi)存(RAM)中是分散的存儲著
  • 地址空間是檢索系統(tǒng)
  • 指針就被編碼為內(nèi)存地址,使用 usize 類型的整數(shù)表示。
    • 一個地址就會指向地址空間中的某個地方
  • 地址空間的范圍是 OS 和 CPU 提供的外觀界面
    • 程序只知道有序的字節(jié)序列,不會考慮系統(tǒng)中實際 RAM 的數(shù)量

名詞解釋

  • 內(nèi)存地址(地址),就是指代內(nèi)存中單個字節(jié)的一個數(shù)
    • 內(nèi)存地址是匯編語言提供的抽象
  • 指針(有時擴展稱為原始指針),就是指向某種類型的一個內(nèi)存地址
    • 指針是高級語言提供的抽象
  • 引用,就是指針。如果是動態(tài)大小的類型,就是指針和具有額外保證的一個整數(shù)
    • 引用是 Rust 提供的抽象

Rust 的引用

  • 引用始終引用的是有效數(shù)據(jù)
  • 引用與 usize 的倍數(shù)對齊
  • 引用可以為動態(tài)大小的類型提供上述保障

Rust 的引用 和 指針

static B: [u8; 10] = [99, 97, 114, 114, 121, 116, 111, 119, 101, 108];
static C: [u8; 11] = [116, 104, 97, 110, 107, 115, 102, 105, 115, 104, 0];
fn main() {
    let a = 42;
    let b = &B;
    let c = &C;
    println!("a: {}, b: {:p}, c: {:p}", a, b, c);
}

運行

point_demo on  master [?] is ?? 0.1.0 via ?? 1.67.1 via ?? base 
? cargo run           
   Compiling point_demo v0.1.0 (/Users/qiaopengjun/rust/point_demo)
    Finished dev [unoptimized + debuginfo] target(s) in 0.44s
     Running `target/debug/point_demo`
a: 42, b: 0x1023dc660, c: 0x1023dc66a

point_demo on  master [?] is ?? 0.1.0 via ?? 1.67.1 via ?? base 

  • 一個更加逼真的例子
    • 使用更復雜的類型展示指針內(nèi)部的區(qū)別
use std::mem::size_of;
static B: [u8; 10] = [99, 97, 114, 114, 121, 116, 111, 119, 101, 108];
static C: [u8; 11] = [116, 104, 97, 110, 107, 115, 102, 105, 115, 104, 0];
fn main() {
    // let a = 42;
    // let b = &B;
    // let c = &C;
    // println!("a: {}, b: {:p}, c: {:p}", a, b, c);
    let a: usize = 42;
    let b: Box<[u8]> = Box::new(B);
    let c: &[u8; 11] = &C;
    println!("a (unsigned 整數(shù)):");
    println!("  地址: {:p}", &a);
    println!("  大小:    {:?} bytes", size_of::<usize>());
    println!("  值:  {:?}\n", a);
    println!("b (B 裝在 Box 里):");
    println!("  地址:  {:p}", &b);
    println!("  大小:    {:?} bytes", size_of::<Box<[u8]>>());
    println!("  指向:  {:p}\n", b);
    println!("c (C 的引用):");
    println!("  地址:  {:p}", &c);
    println!("  大小:  {:?} bytes", size_of::<&[u8; 11]>());
    println!("  指向:  {:p}\n", c);
    println!("B (10 bytes 的數(shù)組):");
    println!("  地址:  {:p}", &B);
    println!("  大小:  {:?} bytes", size_of::<[u8; 10]>());
    println!("  值:  {:?}\n", B);
    println!("C (11 bytes 的數(shù)字):");
    println!("  地址:  {:p}", &C);
    println!("  大小:  {:?} bytes", size_of::<[u8; 11]>());
    println!("  值:  {:?}\n", C);
}

運行

point_demo on  master [?] is ?? 0.1.0 via ?? 1.67.1 via ?? base 
? cargo run
   Compiling point_demo v0.1.0 (/Users/qiaopengjun/rust/point_demo)
    Finished dev [unoptimized + debuginfo] target(s) in 0.19s
     Running `target/debug/point_demo`
a (unsigned 整數(shù)):
  地址: 0x16dda9a08
  大小:    8 bytes
  值:  42

b (B 裝在 Box 里):
  地址:  0x16dda9a10
  大小:    16 bytes
  指向:  0x12b606ba0

c (C 的引用):
  地址:  0x16dda9a30
  大小:  8 bytes
  指向:  0x10208d7ba

B (10 bytes 的數(shù)組):
  地址:  0x10208d7b0
  大小:  10 bytes
  值:  [99, 97, 114, 114, 121, 116, 111, 119, 101, 108]

C (11 bytes 的數(shù)字):
  地址:  0x10208d7ba
  大小:  11 bytes
  值:  [116, 104, 97, 110, 107, 115, 102, 105, 115, 104, 0]

point_demo on  master [?] is ?? 0.1.0 via ?? 1.67.1 via ?? base 

  • 對 B 和 C 中文本進行解碼的例子
    • 它創(chuàng)建了一個與前圖更加相似的內(nèi)存地址布局
use std::borrow::Cow;
use std::ffi::CStr;
use std::os::raw::c_char;
static B: [u8; 10] = [99, 97, 114, 114, 121, 116, 111, 119, 101, 108];
static C: [u8; 11] = [116, 104, 97, 110, 107, 115, 102, 105, 115, 104, 0];
fn main() {
  let a = 42;
  let b: String;
  let c: Cow<str>;
  unsafe {
    let b_ptr = &B as * const u8 as *mut u8;
    b = String::from_raw_parts(b_ptr, 10, 10);
    let c_ptr = &C as *const u8 as *const c_char;
    c = CStr::from_ptr(c_ptr).to_string_lossy();
  }
  println!("a: {}, b: {}, c: {}", a, b, c);
}

Raw Pointers(原始指針)

  • Raw Pointer (原始指針)是沒有 Rust 標準保障的內(nèi)存地址。
    • 這些本質上是 unsafe 的
  • 語法:
    • 不可變 Raw Pointer:*const T
    • 可變的 Raw Pointer:*mut T
    • 注意:*const T,這三個標記放在一起表示的是一個類型
    • 例子:*const String
  • *const T 與 *mut T 之間的差異很小,相互可以自由轉換
  • Rust 的引用(&mut T 和 &T)會編譯為原始指針
    • 這意味著無需冒險進入 unsafe 塊,就可以獲得原始指針的性能
  • 例子:把引用轉為原始指針
fn main() {
  let a: i64 = 42;
  let a_ptr = &a as *const i64;
  println!("a: {} ({:p})", a, a_ptr);
}
  • 解引用(dereference):通過指針從 RAM 內(nèi)存提取數(shù)據(jù)的過程叫做對指針進行解引用(dereferencing a pointer)
  • 例子:把引用轉為原始指針
fn main() {
  let a: i64 = 42;
  let a_ptr = &a as *const i64;
  let a_addr: usize = unsafe {std::mem::transmute(a_ptr)};
  println!("a: {} ({:p}...0x{:x})", a, a_ptr, a_addr + 7);
}

關于 Raw Pointer 的提醒

  • 在底層,引用(&T 和 &mutT)被實現(xiàn)為原始指針。但引用帶有額外的保障,應該始終作為首選使用
  • 訪問 Raw Pointer 的值總是 unsafe 的
  • Raw Pointer 不擁有值的所有權
    • 在訪問時編譯器不會檢查數(shù)據(jù)的合法性
  • 允許多個 Raw Pointer 指向同一數(shù)據(jù)
    • Rust 無法保證共享數(shù)據(jù)的合法性

使用 Raw Pointer 的情況

  • 不可避免
    • 某些 OS 或 第三方庫需要使用,例如與C交互
  • 共享對某些內(nèi)容的訪問至關重要,運行時性能要求高

Rust 指針生態(tài)

  • Raw Pointer 是 unsafe 的
  • Smart Pointer(智能指針)傾向于包裝原始指針,附加更多的能力
    • 不僅僅是對內(nèi)存地址解引用

Rust 智能指針

名稱簡介強項弱項
Raw Pointer*mut T 和 *const T,自由基,閃電般塊,極其 Unsafe速度、與外界交互Unsafe
Box<T>可把任何東西都放在Box里。可接受幾乎任何類型的長期存儲。新的安全編程時代的主力軍。將值集中存儲在 Heap大小增加
Rc<T>是Rust的能干而吝嗇的簿記員。它知道誰借了什么,何時借了什么對值的共享訪問大小增加;運行時成本;線程不安全
Arc<T>是Rust的大使。它可以跨線程共享值,保證這些值不會相互干擾對值的共享訪問;線程安全大小增加;運行時成本
Cell<T>變態(tài)專家,具有改變不可變值的能力內(nèi)部可變性大小增加;性能
RefCell<T>對不可變引用執(zhí)行改變,但有代價內(nèi)部可變性;可與僅接受不可變引用的Rc、Arc嵌套使用大小增加;運行時成本;缺乏編譯時保障
Cow<T>封閉并提供對借用數(shù)據(jù)的不可變訪問,并在需要修改或所有權時延遲克隆數(shù)據(jù)當只是只讀訪問時避免寫入大小可能會增大
String可處理可變長度的文本,展示了如何構建安全的抽象動態(tài)按需增長;在運行時保證正確編碼過度分配內(nèi)存大小
Vec<T>程序最常用的存儲系統(tǒng);它在創(chuàng)建和銷毀值時保持數(shù)據(jù)有序動態(tài)按需增長過度分配內(nèi)存大小
RawVec<T>Vec<T>和其它動態(tài)大小類型的基石;知道如何按需給你的數(shù)據(jù)提供一個家動態(tài)按需增長;與內(nèi)存分配器一起配合尋找空間不直接適用于您的代碼
Unique<T>作為值的唯一所有者,可保證擁有完全控制權需要獨占值的類型(如 String)的基礎不適合直接用于應用程序代碼
Shared<T>分享所有權很難,但他使生活更輕松共享所有權;可以將內(nèi)存與T的寬度對齊,即使是空的時候不適合直接用于應用程序代碼

到此這篇關于Rust語言中級教程之指針的文章就介紹到這了,更多相關Rust指針內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 深入了解Rust中的枚舉和模式匹配

    深入了解Rust中的枚舉和模式匹配

    這篇文章主要為大家詳細介紹了Rust中的枚舉和模式匹配的相關知識,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2024-01-01
  • Rust中箱、包和模塊的學習筆記

    Rust中箱、包和模塊的學習筆記

    Rust中有三個重要的組織概念:箱、包、模塊,本文主要介紹了Rust中箱、包和模塊的學習筆記,文中通過示例代碼介紹的非常詳細,需要的朋友們下面隨著小編來一起學習學習吧
    2024-03-03
  • Rust 模式匹配示例詳解

    Rust 模式匹配示例詳解

    這篇文章主要為大家介紹了Rust 模式匹配示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-10-10
  • Rust之Substrate框架中Core詳解

    Rust之Substrate框架中Core詳解

    Substrate是一個用于構建區(qū)塊鏈的開發(fā)框架,它由Parity團隊基于Rust語言開發(fā)而成,是一個開箱即用的區(qū)塊鏈構造器,本文詳細介紹了Substrate框架中的Core,需要的朋友可以參考下
    2023-05-05
  • Rust?use關鍵字妙用及模塊內(nèi)容拆分方法

    Rust?use關鍵字妙用及模塊內(nèi)容拆分方法

    這篇文章主要介紹了Rust?use關鍵字妙用|模塊內(nèi)容拆分,文中還給大家介紹use關鍵字的習慣用法,快速引用自定義模塊內(nèi)容或標準庫,以此優(yōu)化代碼書寫,需要的朋友可以參考下
    2022-09-09
  • Rust語言之Prometheus系統(tǒng)監(jiān)控工具包的使用詳解

    Rust語言之Prometheus系統(tǒng)監(jiān)控工具包的使用詳解

    Prometheus?是一個開源的系統(tǒng)監(jiān)控和警報工具包,最初是由SoundCloud構建的,隨著時間的發(fā)展,Prometheus已經(jīng)具有適用于各種使用場景的版本,為了開發(fā)者方便開發(fā),更是有各種語言版本的Prometheus的開發(fā)工具包,本文主要介紹Rust版本的Prometheus開發(fā)工具包
    2023-10-10
  • rust流程控制的具體使用

    rust流程控制的具體使用

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

    詳解Rust中泛型的使用

    泛型是一種在編寫代碼時不指定具體類型,而在使用時再確定類型的編程技巧,本文將詳細介紹?Rust?中泛型的相關概念和用法,希望對大家有所幫助
    2023-06-06
  • rust?中生成與使用protobuf的方法

    rust?中生成與使用protobuf的方法

    這篇文章主要介紹了rust中protobuf生成與使用,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-05-05
  • 關于Rust?使用?dotenv?來設置環(huán)境變量的問題

    關于Rust?使用?dotenv?來設置環(huán)境變量的問題

    在項目中,我們通常需要設置一些環(huán)境變量,用來保存一些憑證或其它數(shù)據(jù),這時我們可以使用dotenv這個crate,接下來通過本文給大家介紹Rust?使用dotenv來設置環(huán)境變量的問題,感興趣的朋友一起看看吧
    2022-01-01

最新評論