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

Rust中字符串類型&str和String的使用

 更新時間:2024年03月14日 09:50:19   作者:希望_睿智  
在Rust中,字符串是一種非常重要的數(shù)據(jù)類型,&str和String是Rust中兩種主要的字符串類型,本文主要介紹了Rust中字符串類型&str和String的使用,感興趣的可以了解一下

概述

在Rust中,字符串是一種非常重要的數(shù)據(jù)類型,用于處理文本數(shù)據(jù)。Rust的字符串是以UTF-8編碼的字節(jié)序列,主要有兩種類型:&str和String。其中,&str是一個對字符數(shù)據(jù)的不可變引用,更像是對現(xiàn)有字符串?dāng)?shù)據(jù)的“視圖”,而String則是一個獨立、可變更的字符串實體。

&str和String

&str和String是Rust中兩種主要的字符串類型,它們在以下6個方面存在比較明顯的區(qū)別。

所有權(quán)和可變性

  • &str:是Rust核心語言中唯一的字符串類型,它是一個不可變的字符串切片,是對字符串?dāng)?shù)據(jù)的引用,并不擁有數(shù)據(jù)的所有權(quán)。&str可以安全地使用,但它的內(nèi)容是不可變的,也就是說,不能改變它指向的字符串的內(nèi)容。&str可以指向String的內(nèi)容,也可以指向靜態(tài)字符串字面量。
  • String:這是一個在堆上分配的、可變的字符串類型。String類型由Rust標(biāo)準(zhǔn)庫提供,而不是編入核心語言。它擁有其內(nèi)容的所有權(quán),這意味著String可以被修改。String本質(zhì)上是一個封裝了動態(tài)大小數(shù)組(Vec<u8>)的結(jié)構(gòu)體,該數(shù)組存儲了UTF-8編碼的字節(jié)。

生命周期

  • &str:生命周期取決于它的來源。如果是字符串字面量,則生命周期為'static。如果來自某個作用域內(nèi)的String或其他類型,則其生命周期與該作用域相同。
  • String:沒有明確的生命周期限制,只要String實例存在,它就可以被使用。

存儲位置

  • &str:可能是指向靜態(tài)內(nèi)存中的字符串字面量(&'static str),比如:編譯時確定的常量字符串。也可能是指向堆上分配的String的一部分,或者任何其他類型的UTF-8編碼數(shù)據(jù)的區(qū)域。
  • String:始終在堆上動態(tài)分配。

性能

  • &str:由于它只是一個引用,沒有額外的內(nèi)存分配成本,因此在某些情況下可能更高效。
  • String:由于它在堆上分配,因此會有額外的內(nèi)存分配和復(fù)制成本,尤其是在字符串拼接時。

使用場景

  • &str:當(dāng)只需要讀取字符串內(nèi)容,或者想要避免額外的內(nèi)存分配時,使用&str。此外,在函數(shù)參數(shù)中,使用&str可以允許函數(shù)接受不同類型的字符串參數(shù),包括:String和靜態(tài)字符串字面量。
  • String:當(dāng)需要一個可變的字符串,或者不關(guān)心字符串的具體來源時,使用String。

與C/C++語言的比較

  • &str:類似于C語言中的const char *,它只是一個指向字符串?dāng)?shù)據(jù)的指針,并不擁有數(shù)據(jù)。在Rust中,&str比C語言中的裸指針更安全,因為它有一個生命周期參數(shù)來確保引用的有效性。
  • String:類似于C++中的std::string,是一個字符的容器,并且擁有其內(nèi)容。

字符串的創(chuàng)建

在Rust中,創(chuàng)建字符串有多種方法。根據(jù)具體需求,我們可以選擇不同的方法。如果需要一個可變的字符串并且打算在程序運行時修改它,那么String類型是最佳選擇。如果只是需要一個對靜態(tài)文本的引用,那么&str就足夠了。

使用字符串字面量創(chuàng)建&str

字符串字面量是在代碼中直接寫入的文本,它們被存儲在程序的只讀數(shù)據(jù)段中,并且是不可變的。字符串字面量隱式地具有&str類型。在下面的示例代碼中,text是一個指向字符串字面量的引用,其類型為&str。

let text: &str = "Hello, CSDN";

使用String::new創(chuàng)建空的String

如果我們想要一個可變的、可以增長的字符串,應(yīng)該使用String類型。在下面的示例代碼中,empty_str是一個空的String變量,我們可以向其中添加內(nèi)容。

fn main() {
    let mut empty_str = String::new();
    empty_str.push_str("Hello");
    println!("{}", empty_str);
}

使用字符串字面量初始化String

可以直接將字符串字面量轉(zhuǎn)換為String,這是通過調(diào)用to_string方法或to_owned方法來實現(xiàn)的。

fn main() {
    let text1 = "Hello, CSDN".to_string();
    let str_slice: &str = "Hello, Rust";
    let text2 = str_slice.to_owned();
    println!("{}", text1);
    println!("{}", text2);
}

使用format!宏創(chuàng)建String

format!宏是Rust中創(chuàng)建格式化字符串的強大工具,它可以根據(jù)提供的格式字符串和參數(shù)生成一個 String。

fn main() {
    let name: &str = "CSDN";
    let info = format!("Hello, {}", name);
    println!("{}", info);
}

使用String::from創(chuàng)建String

String::from是一個便利的方法,用于從實現(xiàn)了Into<String>特征的任何類型創(chuàng)建String。因為字符串字面量隱式地實現(xiàn)了這個特征,故可以直接使用。

let text = String::from("Hello, CSDN");

字符串的拼接

Rust提供了強大的字符串拼接功能,可以讓字符串操作變得更加靈活和高效。

使用+運算符或+=運算符

如果想要將兩個String類型進行拼接,可以使用+運算法。

fn main() {
    let str1 = String::from("Hello");
    let str2 = String::from(" CSDN");
    // 不能直接使用str1 + str2
    let str = str1 + &str2;
    println!("{}", str);
    // 編譯錯誤:value borrowed here after move
    println!("{}", str1);
}

在上面的示例代碼中,我們將str1和str2進行了拼接,并得到了str。拼接時,我們使用了&str2,而沒有直接使用str2。拼接完成后,str1不再有效。之所以會這樣,與使用+運算符時調(diào)用的函數(shù)簽名有關(guān)。Rust的+運算符使用了add函數(shù),其簽名與下面的函數(shù)聲明類似。

fn add(self, s: &str) -> String

首先,str2使用了&,意味著我們使用第二個字符串的引用與第一個字符串相加。這是因為add函數(shù)只能將&str和String相加,而不能將兩個String值相加。在Rust中,可以通過Deref強制轉(zhuǎn)換將&String強轉(zhuǎn)成&str,相當(dāng)于自動把&str2變成了&str2[..]。其次,add函數(shù)直接獲取了self的所有權(quán),因為self沒有使用&。這意味著,str1的所有權(quán)被移動到add函數(shù)后,str1將不再有效。

若要對可變的String進行拼接操作,還可以使用+=操作符。但實際上,這并不是簡單的連接,而是創(chuàng)建了一個新的String實例,并丟棄了原String分配的內(nèi)存。

fn main() {
    let mut str1 = String::from("Hello");
    let str2 = " CSDN";
    str1 += str2;
    println!("{}", str1);
}

注意:使用+=運算符,或者連續(xù)使用+運算符進行多次拼接,會導(dǎo)致多次內(nèi)存分配,效率較低,尤其是在處理大量數(shù)據(jù)時。如果需要高效地拼接多個字符串,建議使用下面的format!宏。

使用format!宏

format!宏是一種更靈活且高效的字符串拼接方法,尤其適用于包含變量和格式化文本的情況。format!宏可以處理各種復(fù)雜的格式化需求,并且它的性能通常優(yōu)于簡單的+拼接。

fn main() {
    let name: &str = "CSDN";
    let info = format!("Hello, {}", name);
    println!("{}", info);
}

使用push_str方法或push方法

如果已經(jīng)有了一個String變量,并且想要將另一個字符串或字符追加到它后面,可以使用push_str方法或push方法。注意:push系列方法不會創(chuàng)建新的String實例,而是直接在原有的String緩沖區(qū)上追加內(nèi)容,這通常比使用+運算符更高效。

fn main() {
    let mut text = String::from("Hello ");
    text.push_str("Rust");
    println!("{}", text);

    text.push(' ');
    text.push('C');
    text.push('S');
    text.push('D');
    text.push('N');
    println!("{}", text);
}

字符串的搜索與替換

在Rust中,我們可以使用find、rfind、contains、replace等方法來進行字符串的搜索與替換。在下面的示例代碼中,我們首先調(diào)用find方法查找子串"CSDN",并返回一個Option類型的值。接下來,我們調(diào)用contains方法來檢查text字符串是否包含了子串"Hello",若包含,返回true,否則返回false。最后,我們調(diào)用replace方法來替換字符串中的子串。

replace方法接收兩個參數(shù):第一個參數(shù)是要被替換的子串,第二個參數(shù)是替換后的新子串。該方法會返回一個新的字符串,其中所有與給定模式匹配的子串都被替換為指定的替換字符串。注意:第一個參數(shù)中的原始字符串不會被修改。

fn main() {
    let text = "Hello CSDN";

    // 搜索子串
    let index = text.find("CSDN");
    if let Some(value) = index {
        println!("found: {}", value);
    } else {
        println!("not found");
    }

    // 包含子串
    let contain_hello = text.contains("Hello");
    println!("contain hello: {}", contain_hello);
    
    // 替換子串
    let replaced: String = text.replace("CSDN", "GitHub");
    println!("{}", replaced);
}

字符串的長度

在Rust中,獲取字符串的長度是一個常見的操作。Rust的String類型提供了一個len方法,可以用來獲取字符串中字節(jié)的數(shù)量。需要特別注意的是:這個長度是以字節(jié)為單位的,對于ASCII字符串來說,每個字符占用一個字節(jié);但是,對于包含多字節(jié)字符(比如:UTF-8編碼的Unicode字符)的字符串,len方法返回的是字節(jié)的總數(shù),而不是字符的總數(shù)。

如果想要獲取字符串中Unicode字符的數(shù)量,我們應(yīng)該使用chars方法,然后計算迭代器中元素的數(shù)量。chars方法會返回一個迭代器,該迭代器逐個產(chǎn)生字符串中的Unicode字符。

fn main() {
    let text = "Hello 霸都";

    // 獲取字節(jié)長度
    let byte_len = text.len();
    // 輸出:12
    println!("{}", byte_len);

    // 獲取字符長度
    let char_len = text.chars().count();
    // 輸出:8
    println!("{}", char_len);
}

另外,Rust字符串不支持直接通過索引來訪問單個字符。這是因為,UTF-8編碼格式下,單個字符可能占用1到4個字節(jié),索引操作會帶來潛在的非確定性和不一致性問題。如果確實需要通過索引訪問字符,可以使用chars()方法。它會返回一個迭代器,產(chǎn)生字符串中的每個Unicode字符。然后,我們可以使用nth方法或者其他集合方法來獲取特定位置的字符。

fn main() {
    let text = "Hello 霸都";
    // 注意:索引從0開始計數(shù)
    let index = 6;
    let cur_char = text.chars().nth(index);
    // 輸出:index 6: 霸
    match cur_char {
        Some(c) => println!("index {}: {}", index, c),
        None => println!("index out of bounds"),
    }
}

字符串與字節(jié)的轉(zhuǎn)換

Rust中的字符串和字節(jié)之間可以方便地進行轉(zhuǎn)換,這在處理二進制數(shù)據(jù)和編解碼時非常有用。

fn main() {
    let text = "Hello CSDN";

    // 字符串轉(zhuǎn)字節(jié)
    let bytes = text.as_bytes();
    // 輸出:[72, 101, 108, 108, 111, 32, 67, 83, 68, 78]
    println!("{:?}", bytes);
    
    // 字節(jié)轉(zhuǎn)字符串
    let bytes2 = [72, 101, 108, 108, 111];
    let text2 = std::str::from_utf8(&bytes2).unwrap();
    // 輸出:Hello
    println!("{}", text2);
}

總結(jié)

由于Rust強調(diào)安全性與內(nèi)存管理,它的字符串設(shè)計也體現(xiàn)出了這一點:不可變的&str確保了引用安全,而String則通過所有權(quán)系統(tǒng)保證了內(nèi)存的有效管理,避免了懸垂引用和其他常見的內(nèi)存錯誤。

到此這篇關(guān)于Rust中字符串類型&str和String的使用的文章就介紹到這了,更多相關(guān)Rust &str String內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Rust語言之trait中的個方法可以重寫嗎

    Rust語言之trait中的個方法可以重寫嗎

    在Rust中,trait定義了一組方法,這些方法可以被一個或多個類型實現(xiàn),當(dāng)你為某個類型實現(xiàn)一個trait時,你可以為該trait中的每個方法提供自己的具體實現(xiàn),本文將給大家介紹一下trait中的個方法是否可以重寫,需要的朋友可以參考下
    2023-10-10
  • Rust使用Channel實現(xiàn)跨線程傳遞數(shù)據(jù)

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

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

    詳解Rust編程中的共享狀態(tài)并發(fā)執(zhí)行

    雖然消息傳遞是一個很好的處理并發(fā)的方式,但并不是唯一一個,另一種方式是讓多個線程擁有相同的共享數(shù)據(jù),本文給大家介紹Rust編程中的共享狀態(tài)并發(fā)執(zhí)行,感興趣的朋友一起看看吧
    2023-11-11
  • Rust實現(xiàn)AES加解密詳解

    Rust實現(xiàn)AES加解密詳解

    這篇文章主要為大家詳細(xì)介紹了如何利用Rust語言實現(xiàn)AES加解密算法,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,需要的可以了解一下
    2022-10-10
  • R語言ggplot2繪圖安裝與調(diào)試

    R語言ggplot2繪圖安裝與調(diào)試

    ggplot2是R語言中最常用的繪圖包之一,它提供了一種基于圖層的繪圖語法,使得用戶可以輕松地創(chuàng)建高質(zhì)量的數(shù)據(jù)可視化圖表。在使用ggplot2之前,需要先安裝該包并進行調(diào)試。安裝ggplot2可以通過CRAN或GitHub進行,調(diào)試則需要注意數(shù)據(jù)格式、語法錯誤等問題。
    2023-06-06
  • Rust應(yīng)用調(diào)用C語言動態(tài)庫的操作方法

    Rust應(yīng)用調(diào)用C語言動態(tài)庫的操作方法

    這篇文章主要介紹了Rust應(yīng)用調(diào)用C語言動態(tài)庫,本文記錄了筆者編寫一個簡單的C語言動態(tài)庫,并通過Rust調(diào)用動態(tài)庫導(dǎo)出的函數(shù),需要的朋友可以參考下
    2023-01-01
  • rust交叉編譯問題及報錯解析

    rust交叉編譯問題及報錯解析

    這篇文章主要為大家介紹了rust交叉編譯問題及報錯解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • Rust中non_exhaustive的enum使用確保程序健壯性

    Rust中non_exhaustive的enum使用確保程序健壯性

    這篇文章主要為大家介紹了Rust中non_exhaustive的enum使用確保程序健壯性示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-11-11
  • Rust?中?Deref?Coercion講解

    Rust?中?Deref?Coercion講解

    Rust 的設(shè)計理念一向是顯式比隱式好,也就是說所有的行為盡量在代碼中表現(xiàn)出來,這篇文章主要介紹了Rust?中?Deref?Coercion?介紹,需要的朋友可以參考下
    2022-10-10
  • 聊聊Rust 運算符

    聊聊Rust 運算符

    運算符 用于對數(shù)據(jù)執(zhí)行一些操作。被運算符執(zhí)行操作的數(shù)據(jù)我們稱之為操作數(shù)。下面通過本文給大家介紹Rust 運算符的相關(guān)知識,感興趣的朋友一起看看吧
    2021-11-11

最新評論