0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

實(shí)現(xiàn)Rust Trait類型 那么該類型的引用也實(shí)現(xiàn)了trait嗎?

jf_wN0SrCdH ? 來(lái)源:coding到燈火闌珊 ? 2023-08-28 15:25 ? 次閱讀

如果你在一個(gè)類型上實(shí)現(xiàn)了一個(gè)trait,然后引用了這個(gè)類型,那么類型的引用也實(shí)現(xiàn)了這個(gè)trait嗎?

有一段時(shí)間我是這么想的!但實(shí)際上并不是,有時(shí)候Rust為你做的事情可能會(huì)混淆幕后真正發(fā)生的事情。

為了演示,讓我從一個(gè)名為Speaker的trait和一個(gè)實(shí)現(xiàn)該trait的空struct開始:

///定義一個(gè)trait,有一個(gè)speak方法。
traitSpeaker{
fnspeak(&self);
}

/// BasicSpeaker是一個(gè)空結(jié)構(gòu)體,只是為了實(shí)現(xiàn)Speaker。
structBasicSpeaker;

///BasicSpeakers實(shí)現(xiàn)speak方法
implSpeakerforBasicSpeaker{
fnspeak(&self){
println!("Hello!");
}
}
現(xiàn)在,在main函數(shù)中,以下代碼應(yīng)該可以工作:
//創(chuàng)建BasicSpeaker結(jié)構(gòu)體
letspeaker=BasicSpeaker;
//調(diào)用在BasicSpeaker上定義的speak方法
speaker.speak();

確實(shí)如此,它就會(huì)輸出“Hello!”。

如果我引用了一個(gè)BasicSpeaker,我仍然可以對(duì)它調(diào)用speak,因?yàn)镽ust會(huì)自動(dòng)解除對(duì)變量的引用。所以下面的代碼也可以工作:

//定義一個(gè)BasicSpeaker的引用
letspeaker_ref:&BasicSpeaker=&speaker;
//通過(guò)引用調(diào)用在BasicSpeaker上定義的speak方法
speaker_ref.speak();

這可能會(huì)讓你認(rèn)為BasicSpeaker實(shí)現(xiàn)了Speaker,引用&BasicSpeaker也實(shí)現(xiàn)了Speaker。也許是Rust的魔法?

讓我們更具體地測(cè)試一下,定義一個(gè)接受impl Speaker類型形參的函數(shù)。

fnspeak_to(s:implSpeaker){
s.speak();
}

fnmain(){
//創(chuàng)建BasicSpeaker結(jié)構(gòu)體
letspeaker=BasicSpeaker;
//將speaker傳遞給新函數(shù)
speak_to(speaker);
}

這是可行的,因?yàn)锽asicSpeaker實(shí)現(xiàn)了Speaker特性。

讓我們嘗試同樣的事情,但這次是傳遞BasicSpeaker的引用:

//定義一個(gè)BasicSpeaker的引用
letspeaker_ref:&BasicSpeaker=&speaker;
//將引用傳遞給'speak_to'
speak_to(speaker_ref);
這行不通!錯(cuò)誤信息如下所示:
error[E0277]:thetraitbound`&BasicSpeaker:Speaker`isnotsatisfied
-->src/main.rs:31:14
|
31|speak_to(speaker_ref);
|--------^^^^^^^^^^^thetrait`Speaker`isnotimplementedfor`&BasicSpeaker`
||
|requiredbyaboundintroducedbythiscall
|
=help:thetrait`Speaker`isimplementedfor`BasicSpeaker`
note:requiredbyaboundin`speak_to`
-->src/main.rs:16:21
|
16|fnspeak_to(s:implSpeaker){
|^^^^^^^requiredbythisboundin`speak_to`

Formoreinformationaboutthiserror,try`rustc--explainE0277`.

最初的錯(cuò)誤消息是模糊的,但是第一個(gè)代碼塊旁邊的消息更清晰:“&BasicSpeaker沒(méi)有實(shí)現(xiàn)trait Speaker”。

前面的代碼示例演示了你可以在引用上調(diào)用沒(méi)有在該引用上實(shí)現(xiàn)的方法,因?yàn)镽ust會(huì)默默地為你解引用該值。Rust是這樣做的:

//Rust將'speaker_ref.speak()'轉(zhuǎn)換為
(*speaker_ref).speak();

這并不意味著&BasicSpeaker(一個(gè)引用)實(shí)現(xiàn)了Speaker。

直接的解決方案

最直接的解決方案是在BasicSpeaker的引用上實(shí)現(xiàn)Speaker,如下所示:

implSpeakerfor&BasicSpeaker{
fnspeak(&self){
println!("Hello!");
}
}
將其添加到代碼中后,就可以編譯和運(yùn)行了。所以這是一種解決方案,但這并不理想。首先,這基本上是先前實(shí)現(xiàn)的重復(fù)代碼。下面是一個(gè)稍微改進(jìn)的實(shí)現(xiàn),它只調(diào)用底層結(jié)構(gòu)體的方法:
implSpeakerfor&BasicSpeaker{
fnspeak(&self){
return(**self).speak();
}
}

很明顯,我必須對(duì)self進(jìn)行兩次解引用,因?yàn)樵摵瘮?shù)接受&self,而self是&BasicSpeaker。這意味著參數(shù)是一個(gè)&&BasicSpeaker,必須對(duì)其進(jìn)行兩次解引用才能獲得實(shí)現(xiàn)speak()的BasicSpeaker。

好了,現(xiàn)在沒(méi)有那么多代碼復(fù)制了,但是還有另一個(gè)問(wèn)題,如果我想定義另一個(gè)Speaker,比如NamedSpeaker,那么我必須編寫兩次代碼——一次為NamedSpeaker,一次為&NamedSpeaker。

用泛型Trait解決這個(gè)問(wèn)題

我可以寫一個(gè)泛型的實(shí)現(xiàn):“如果一個(gè)struct T實(shí)現(xiàn)了Speaker,那么寫一個(gè)用于&T的Speaker實(shí)現(xiàn)?!?/p>

///所有實(shí)現(xiàn)Speaker的事物的引用也必須是Speaker的。
implSpeakerfor&T
where
T:Speaker,
{
fnspeak(&self){
return(**self).speak();
}
}
或者,如果你喜歡,你可以使用下面的,稍微短一點(diǎn)的語(yǔ)法,意思是一樣的:
implSpeakerfor&T{
fnspeak(&self){
return(**self).speak();
}
}
即使我現(xiàn)在已經(jīng)為&BasicSpeaker編寫了Speaker的實(shí)現(xiàn),但這并不適用于&mut BasicSpeaker!所以這行不通:
//獲取一個(gè)對(duì)BasicSpeaker的可變引用
letspeaker_mut_ref:&mutBasicSpeaker=&mutspeaker;
//通過(guò)可變引用,調(diào)用在BasicSpeaker上定義的speak方法
speak_to(speaker_mut_ref);
這需要另一個(gè)泛型實(shí)現(xiàn):
///所有實(shí)現(xiàn)Speaker的事物的可變引用也必須是Speaker的。
implSpeakerfor&mutT
where
T:Speaker,
{
fnspeak(&self){
return(**self).speak();
}
}
為了完整起見,這里對(duì)于Box也是一樣的,當(dāng)你想把Speaker實(shí)現(xiàn)放到堆上時(shí):
implSpeakerforBox
where
T:Speaker,
{
fnspeak(&self){
return(**self).speak();
}
}
一旦添加了這些覆蓋實(shí)現(xiàn),就意味著Speaker的任何新類型(該類型本身、對(duì)該類型的任何引用以及包含該類型的任何Box)都自動(dòng)實(shí)現(xiàn)了Speaker trait。

總結(jié)

因?yàn)镽ust會(huì)自動(dòng)解除對(duì)trait的引用,它看起來(lái)就像引用本身也實(shí)現(xiàn)了trait。但事實(shí)并非如此。幸運(yùn)的是,在許多情況下,你可以使用一些泛型trait來(lái)修復(fù)這個(gè)問(wèn)題。

如果你的trait接口允許,你應(yīng)該為&T, &mut T和Box提供trait實(shí)現(xiàn),這樣你就可以將這些類型傳遞給任何接受trait實(shí)現(xiàn)的函數(shù)。






審核編輯:劉清

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • rust語(yǔ)言
    +關(guān)注

    關(guān)注

    0

    文章

    57

    瀏覽量

    3009

原文標(biāo)題:實(shí)現(xiàn)了Rust Trait的類型,那么該類型的引用也實(shí)現(xiàn)了trait嗎?

文章出處:【微信號(hào):Rust語(yǔ)言中文社區(qū),微信公眾號(hào):Rust語(yǔ)言中文社區(qū)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    基于Rust語(yǔ)言Hash特征的基礎(chǔ)用法和進(jìn)階用法

    Rust語(yǔ)言是一種系統(tǒng)級(jí)編程語(yǔ)言,具有高性能、安全、并發(fā)等特點(diǎn),是近年來(lái)備受關(guān)注的新興編程語(yǔ)言。在Rust語(yǔ)言中,Hash是一種常用的數(shù)據(jù)結(jié)構(gòu),用于存儲(chǔ)鍵值對(duì)。Rust語(yǔ)言提供一系列
    的頭像 發(fā)表于 09-19 16:02 ?1472次閱讀

    Rust語(yǔ)言中的反射機(jī)制

    Rust語(yǔ)言的反射機(jī)制指的是在程序運(yùn)行時(shí)獲取類型信息、變量信息等的能力。Rust語(yǔ)言中的反射機(jī)制主要通過(guò) Any 實(shí)現(xiàn)。 std::any::Any
    的頭像 發(fā)表于 09-19 16:11 ?2454次閱讀

    Rust中的From和Into trait的基礎(chǔ)使用方法和進(jìn)階用法

    、可靠和安全的系統(tǒng)級(jí)應(yīng)用的首選語(yǔ)言。 Rust中的From和Into是兩個(gè)重要的trait,它們可以幫助我們進(jìn)行類型轉(zhuǎn)換。From trait允許我們從一個(gè)
    的頭像 發(fā)表于 09-20 10:55 ?1802次閱讀

    “布爾2”的引用 和“Bool引用句柄”為什么不是同一個(gè)數(shù)據(jù)類型?

    請(qǐng)教 改變了 按鈕的機(jī)械方式, 右圖中“布爾2”的引用 和“Bool引用句柄”為什么不是同一個(gè)數(shù)據(jù)類型?
    發(fā)表于 04-04 23:21

    子vi如何使其變?yōu)榉菄?yán)格類型

    請(qǐng)教各位大神一個(gè)問(wèn)題,在一個(gè)vi中調(diào)用子vi時(shí),通過(guò)“右鍵>選擇服務(wù)器類型>瀏覽”方式選擇一個(gè)子vi后其為嚴(yán)格類型,要想實(shí)現(xiàn)引用還得使用下
    發(fā)表于 05-04 20:29

    Windows -編程-數(shù)據(jù)類型

    Windows -編程-數(shù)據(jù)類型Rust 中的每個(gè)值都有特定的數(shù)據(jù)類型,它告訴 Rust 指定什么樣的數(shù)據(jù),以便它知道如何處理這些數(shù)據(jù)。我
    發(fā)表于 08-24 14:30

    引用數(shù)據(jù)類型的概念_引用數(shù)據(jù)類型有哪幾種

    引用類型類型的實(shí)際值引用(類似于指針)表示的數(shù)據(jù)類型。如果為某個(gè)變量分配一個(gè)引用
    發(fā)表于 11-17 18:29 ?2.6w次閱讀
    <b class='flag-5'>引用</b>數(shù)據(jù)<b class='flag-5'>類型</b>的概念_<b class='flag-5'>引用</b>數(shù)據(jù)<b class='flag-5'>類型</b>有哪幾種

    C#良好兼容類型/引用類型

    反觀歷史,C#曾經(jīng)因?yàn)?值類型/引用類型 保守詬病,“拆箱”和“裝箱”一直是個(gè)招黑的設(shè)計(jì)。但后來(lái)我們看到,隨著泛型的成熟和普及,隨著泛型容器代替通用容器,裝箱和拆箱的問(wèn)題已經(jīng)在很大程
    的頭像 發(fā)表于 11-20 10:14 ?1703次閱讀
    C#良好兼容<b class='flag-5'>了</b>值<b class='flag-5'>類型</b>/<b class='flag-5'>引用</b><b class='flag-5'>類型</b>

    Rust原子類型和內(nèi)存排序

    原子類型在構(gòu)建無(wú)鎖數(shù)據(jù)結(jié)構(gòu),跨線程共享數(shù)據(jù),線程間同步等多線程并發(fā)編程場(chǎng)景中起到至關(guān)重要的作用。本文將從Rust提供的原子類型和原子類型的內(nèi)存排序問(wèn)題兩方面來(lái)介紹。
    的頭像 發(fā)表于 10-31 09:21 ?957次閱讀

    Rust中GAT和高階類型

    Rust類型系統(tǒng)級(jí)別上與Haskell,Scala有許多相似之處。
    的頭像 發(fā)表于 11-07 10:21 ?1181次閱讀

    trait中使用 `async fn`

    { ... } } 更多請(qǐng)看官方blog:https://blog.rust-lang.org/inside-rust/2022/11/17/async-fn-in-trait-nightly.html 社區(qū)受
    的頭像 發(fā)表于 11-23 15:40 ?790次閱讀

    重點(diǎn)講解Send與Sync相關(guān)的并發(fā)知識(shí)

    Send與Sync在Rust中屬于marker trait,代碼位于marker.rs,在標(biāo)記模塊中還有Copy、Unpin等trait
    的頭像 發(fā)表于 01-16 09:54 ?949次閱讀

    為什么Borrow和BorrowMut被定義為泛型trait呢?

    Borrow和BorrowMut traitRust標(biāo)準(zhǔn)庫(kù)std:borrow 模塊中用于處理借用數(shù)據(jù)的trait,通過(guò)實(shí)現(xiàn)Borrow 和BorrowMut
    的頭像 發(fā)表于 05-22 15:57 ?911次閱讀

    rust語(yǔ)言基礎(chǔ)學(xué)習(xí): 智能指針之Cow

    Rust中與借用數(shù)據(jù)相關(guān)的三個(gè)trait: Borrow, BorrowMut和ToOwned。理解了這三個(gè)trait之后,再學(xué)習(xí)Rust中能夠實(shí)現(xiàn)
    的頭像 發(fā)表于 05-22 16:13 ?2933次閱讀

    rust語(yǔ)言基礎(chǔ)學(xué)習(xí): Default trait

    Default trait 顧名思義是默認(rèn)值,即Rust為特定類型實(shí)現(xiàn) Default 特性時(shí),可以為該類型賦予
    的頭像 發(fā)表于 05-22 16:16 ?1287次閱讀