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

淺談Rust?+=?運算符與?MIR?應(yīng)用

 更新時間:2023年01月31日 10:26:54   作者:Mr_Vague  
這篇文章主要介紹了Rust?+=?運算符與?MIR?應(yīng)用,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

+= 運算符與 MIR 應(yīng)用

本文 += 運算符部分整理自 Why does += require manual dereference when AddAssign() does not? 后半部分,
MIR 部分是我自己補充的。

只在 https://zjp-cn.github.io/rust-note/ 上更新,其他地方懶得同步更新。

+= 解語法糖

一個基礎(chǔ),但很少會思考的問題,Rust 的 += 運算符是什么代碼的語法糖?

a = a + b 不等價于 a += b

a = a + ba += b 的語法糖嗎?這意味著任何 a += b 與任何 a = a + b 代碼等價。

如果以標(biāo)準庫定義的 impls 為例子,你可能覺得兩種寫法都能編譯,而且結(jié)果一致。

但考慮以下自定義類型的實現(xiàn):

use std::ops::{Add, AddAssign};

fn main() {
    let mut s = S;
    s += (); // ok
    s = s + (); // error: expected struct `S`, found `()`
}

struct S;

impl Add<()> for S {
    type Output = ();
    fn add(self, _: ()) { }
}

impl AddAssign<()> for S {
    fn add_assign(&mut self, _: ()) { }
}

代碼不通過,原因是顯然的,s + () 的類型是 (),無法賦值給 s —— a = a + b 不是 a += b 的語法糖。

從運算符的 trait 定義來看(以 + vs += 為例),它們沒有任何關(guān)系:

pub trait Add<Rhs = Self> {
    type Output;

    fn add(self, rhs: Rhs) -> Self::Output;
}

pub trait AddAssign<Rhs = Self> {
    fn add_assign(&mut self, rhs: Rhs);
}

AddAssign::add_assign(&mut a, b)a += b

++= 是典型的二元運算符和復(fù)合賦值運算符。根據(jù)各自的運算符 trait 定義,可以得到以下解語法糖:

  • a + b 實際調(diào)用 Add::add(a, b)
  • a += b 實際調(diào)用 AddAssign::add_assign(&mut a, b)

注意以下幾點:

  • 若 a 和 b 擁有所有權(quán)時,其右側(cè)運算數(shù) b 的所有權(quán)被獲取1,而對待左側(cè)運算數(shù)所有權(quán)的方式并不相同:
  • a + b 獲取了 a 的所有權(quán)(無法再使用 a)
  • a += b 獲取了 a 的獨占引用,而非所有權(quán)(a 必須是 mut 的,而且此后仍可以使用 a)
  • 若 a 或 b 不擁有所有權(quán)時,則不存在對 a 或 b 所有權(quán)的轉(zhuǎn)移2:
  • 當(dāng) implementor 為引用時,參數(shù)一并沒有發(fā)生所有權(quán)的移動
  • 當(dāng)泛型類型參數(shù)為引用時,參數(shù)二并沒有發(fā)生所有權(quán)的移動
  • 調(diào)用的形式最好使用完全限定語法,而不是方法調(diào)用語法。這是因為方法調(diào)用表達式存在隱式的 自動引用/解引用,而基于類型的分析才更可靠。
  • a + b 實際調(diào)用 <TypeOfA as Add<TypeOfB>>::add(a, b),優(yōu)于 a.add(b)
  • a += b 實際調(diào)用 <TypeOfA as AddAssign<TypeOfB>>::add_assign(&mut a, b),優(yōu)于 (&mut a).add_assign(b)

本文的重點在于 a += b,而不是 a + b,所以對 a + b 的內(nèi)容就此結(jié)束。

對于分析 a += b,我遵循以下思考流程:

  • 寫下兩側(cè)的類型,
  • Self += Rhs實際調(diào)用的形式,如
  • <Self as AddAssign<Rhs>>::add_assign(&mut Self, Rhs)
  • <Self as AddAssign<Rhs>>::add_assign(&mut a, b)
  • 完全限定語法的幾種等價形式:
    • Self: AddAssign<Rhs>:這在分析 trait bounds 時常用
    • impl AddAssign<Rhs> for Self:這在搜索具體實現(xiàn)時有用3

但像 += 這樣的“復(fù)合賦值運算符”,一個鮮為人知的規(guī)則是關(guān)于兩側(cè)運算數(shù)的求值順序。

賦值表達式的求值順序

通過一個示例來感受求值順序為什么重要:

*{
    print!("lhs ");
    &mut 0
} += {
    print!("rhs ");
    0
};

*{
    print!("lhs ");
    &mut String::from("a")
} += {
    print!("rhs ");
    "b"
};

這段代碼打印什么?

如果你能準確說出和解釋打印的內(nèi)容,那么這小節(jié)內(nèi)容可以跳過了。

如果你不知道答案,請往下看。

規(guī)則

Rust 中,大部分表達式是從左往右求值的,比如對于方法調(diào)用表達式 (method call expression) a.add(b),脫糖為
Add::add(a, b),然后先計算左邊的 a,再計算右邊的 b。但賦值表達式不一定是從左到右求值。

Rust 具有兩種“賦值表達式”:

  • 賦值表達式 (assignment expressions):將一個值移動進指定的地方,語法為 assignee operand = assigned value operand。
  • 復(fù)合賦值表達式 (compound assignment expressions):將運算/邏輯二元運算符與賦值表達式結(jié)合起來,語法為
  • assigned operand 操作符 modifying operand,其中“操作符”為一個標(biāo)記后跟一個 =(中間不含空格),比如 +=|=、<<=

兩側(cè)運算數(shù)的名稱非常不直觀,所以我使用左右兩側(cè)的表達方式來稱呼它們。實際上,它們以前被稱作“左值” (lvalue) 和“右值” (rvalue)。

對賦值表達式來說,先計算等號右側(cè)的值,再計算等號左側(cè)的值,即從右到左;對于解構(gòu)賦值,其內(nèi)部求值順序為從左到右。

# let (mut a, mut b);

(a, b) = (3, 4); // 從右到左:先計算等號右側(cè)的 (3, 4),再賦值給等號左側(cè)的 (a, b)

// 脫糖為

{
    let (_a, _b) = (3, 4); // 解構(gòu)賦值過程中,從左到右
    a = _a; // 先賦值給解構(gòu)模式左邊的 a
    b = _b; // 再賦值給解構(gòu)模式右邊的 b
}

對于復(fù)合賦值表達式,若兩側(cè)的類型同時為 primitives,從右到左計算;否則從左到右計算。

回到本小節(jié)開頭的示例,現(xiàn)在可以仔細分析代碼了:

// 等號兩側(cè)的類型都為 `i32`,它是 primitive type,所以從右到左計算,打印 `rhs lhs `
*{
    print!("lhs ");
    &mut 0
} += {
    print!("rhs ");
    0
};

// 等號左右的類型為 `String` 和 `&str`,都不是 primitive type,所以從左到右計算,打印 `lhs rhs `
*{
    print!("lhs ");
    &mut String::from("a")
} += {
    print!("rhs ");
    "b"
};

或許這些細節(jié)你會感到困惑:

等號左側(cè)為什么要那樣寫?因為不允許直接寫 0 += ...
為什么左側(cè)可以維持臨時的引用 &mut?temporary-lifetime-extension
為什么左側(cè)類型是 i320 的類型為 i32,這是 Rust 默認推斷的;&mut 0 類型為 &mut i32;*&mut 0 類型為 i32
為什么 i32 是 primitive type?見標(biāo)準庫 primitive types
什么是 primitive type?見標(biāo)準庫 primitive types
為什么 &str 不是 primitive type?見標(biāo)準庫 primitive types,且見下面的例子:i32 是,&i32 不是,所以 str 是,&str 不是

總而言之,在 Rust 中,大部分表達式的求值順序是從左往右的,僅有少數(shù)地方是從右往左的,比如:

賦值表達式:先計算等號右側(cè)復(fù)合賦值表達式:僅在兩側(cè)運算數(shù)都為 primitive types 時才先計算右側(cè)運算數(shù)。為了鞏固這一條,請確保你完全理解下面的
代碼 和注釋。此外,你還可以看懂 rustc 的這個 測試代碼。

use std::num::Wrapping;

macro_rules! add_assign {
    ($e1:expr, $e2:expr) => {
        *({print!("lhs "); &mut $e1}) += {print!("rhs "); $e2};
        println!("");
    }
}

fn main() {
    add_assign!(1, 2); // rhs lhs: both operands are primitives
    add_assign!(1, &2); // lhs rhs: Rhs &i32 is not a direct primitive
    add_assign!(String::new(), ""); // lhs rhs: neither operands are primitives
    add_assign!(Wrapping(1), Wrapping(2)); // lhs rhs: neither operands are primitives
    // So usually the execution order of `+=` is LTR (left-to-right)
}

MIR

Rust 的 MIR 是 HIR 到 LLVM IR 的中間產(chǎn)物,對 Rust 眾多語法糖進行了脫糖,并且極大地精簡了 Rust
語法(但并非其語法子集),是觀察和分析 Rust 代碼的常用手段,尤其是在控制流圖和借用檢查方面。

獲取 MIR 的最簡便的方式是通過 playground 左上角下拉框,點擊 MIR 按鈕。

此外,你還可以使用 rustc src/main.rs -Z dump-mir=maincargo rustc -- -Z dump-mir=main 獲得有關(guān) main 函數(shù)完整的 MIR

  • 查看 mir_dump/main.main.-------.renumber.0.mir 等文件
  • 使用 cargo rustc -- -Z help 查看更多 mir 相關(guān)命令
  • 相關(guān) MIR 資料

Rust Blog: Introducing MIR 友好的官方入門解釋

rustc-dev-guide: MIR Debugging

rustc-dev-guide: The MIR (Mid-level IR)

對于上一節(jié)開頭的示例:

// 去除了無關(guān)和冗雜的 print!,將這段代碼復(fù)制到 play.rust-lang.org 查看 MIR
*{ &mut 0 } += 0;
*{ &mut String::from("a") } += "b";

關(guān)鍵的 MIR 輸出:

bb0: {
    _1 = const 0_i32;
    _3 = const 0_i32;
    _2 = &mut _3;
    _4 = CheckedAdd((*_2), _1);
    assert(!move (_4.1: bool), "attempt to compute `{} + {}`, which would overflow", (*_2), move _1) -> bb1;
}

bb2: {
    _7 = &mut _8;
    _6 = &mut (*_7);
    _10 = const "b";
    _9 = _10;
    _5 = <String as AddAssign<&str>>::add_assign(move _6, move _9) -> [return: bb3, unwind: bb5];
}

這很容易解釋 += 的語法脫糖和真正的執(zhí)行順序:

  • _4 = CheckedAdd((*_2), _1) 這里的執(zhí)行順序是從右到左(注意觀察編號),并且不是調(diào)用 <i32 as AddAssign<i32>>::add_assign,

而是直接調(diào)用 CheckedAdd 函數(shù)。

  • add_assign!(1, &2) 則對應(yīng) _1 = <i32 as AddAssign<&i32>>::add_assign(move _2, move _13) -> bb5,順序從左到右,調(diào)用了重載的

+= trait 方法。

  • _5 = <String as AddAssign<&str>>::add_assign(move _6, move _9) 這里的順序是從左到右,調(diào)用的是重載的 += trait 方法。

單一實現(xiàn)下的強轉(zhuǎn)

遵循前面我提到的流程,對于以下正常工作代碼,第一步,寫下左右兩側(cè)的類型,你會得到 S += &&&&&&(),實際不存在這個實現(xiàn),因為
S 僅有 S: AddAssign<&()>。這發(fā)生了什么?

struct S;
impl std::ops::AddAssign<&()> for S {
    fn add_assign(&mut self, _: &()) {}
}

fn main() {
    let mut s = S;
    let rrrrrr = &&&&&&();
    s += rrrrrr;
}

通過 MIR,你會發(fā)現(xiàn)

  • <S as AddAssign<&()>>::add_assign(move _4, move _5) 表明從左到右執(zhí)行,因為兩側(cè)運算數(shù)不是 primitive type
  • 傳給 add_assign 的第二個參數(shù),其類型并不是變量 rrrrrr 的類型 &&&&&&(),而是經(jīng)過 5 次解引用之后的 &() 類型
bb0: {
    _6 = const _;
    _4 = &mut _1;
    _7 = deref_copy (*_2);
    _8 = deref_copy (*_7);
    _9 = deref_copy (*_8);
    _10 = deref_copy (*_9);
    _11 = deref_copy (*_10);
    _5 = _11;
    _3 = <S as AddAssign<&()>>::add_assign(move _4, move _5) -> bb1;
}

這里隱式的解引用是因為強轉(zhuǎn),而函數(shù)參數(shù)是能夠發(fā)生 強轉(zhuǎn)的地方 之一。

并且,依據(jù)這段 MIR(注意看從上到下的執(zhí)行過程),我們知道,對于已知的 add_assign 實現(xiàn),執(zhí)行順序先于強轉(zhuǎn)發(fā)生。

而當(dāng) SAddAssign 實現(xiàn)是多個,強轉(zhuǎn)被阻止,你需要傳入準確的類型的值:

struct S;
impl std::ops::AddAssign<()> for S {
    fn add_assign(&mut self, _: ()) {}
}
impl std::ops::AddAssign<&()> for S {
    fn add_assign(&mut self, _: &()) {}
}

fn main() {
    let mut s = S;
    let rrrrrr = &&&&&&();
    s += rrrrrr;
}

// error[E0277]: cannot add-assign `&&&&&&()` to `S`
//   --> src/main.rs:12:7
//    |
// 12 |     s += rrrrrr;
//    |       ^^ no implementation for `S += &&&&&&()`
//    |
//    = help: the trait `AddAssign<&&&&&&()>` is not implemented for `S`
//    = help: the following other types implement trait `AddAssign<Rhs>`:
//              <S as AddAssign<&()>>
//              <S as AddAssign<()>>

兩階段借用的參與

以下代碼能夠運行:

  • 由于兩側(cè)類型不是 primitive type, add_assign 從左到右執(zhí)行
  • 但已經(jīng)使用 &mut self 的情況下,為什么能夠同時執(zhí)行帶 &self 的方法?
struct S;
impl std::ops::AddAssign<()> for S {
    fn add_assign(&mut self, _: ()) {}
}
impl S {
    fn no_op(&self) {}
}

fn main() {
    let mut s = S;
    s += s.no_op();
}

通常對于初學(xué)者, &mut 會有兩個更高級的主題:

  • 重新借用 (reborrow)
  • open 狀態(tài)的 Reference issue、RFC issue,在遷移到 Chalk 之前,不會正式描述 reborrow
  • 它大概是說:我們看見的 &'a mut T,實際被自動轉(zhuǎn)化成更短的 &'b mut T,從而看起來 &mut T 一直可用。這也發(fā)生在
  • &T 上面,但通常我們對 &mut T 的 reborrow 更敏感。
  • 這一是個在 1.0 之前就有的概念
  • UCG 可能會對 reborrow 做出說明
  • 一個直覺上的理解
  • 兩階段借用 (two-phase borrows)
  • 它在 rustc dev guide 上的 正式介紹
  • 它大概是說,某些情況下 &mut T 會劃分成兩個階段進行使用:
  • 在 reservation 階段:&mut T 像是 &T 那樣,以允許多個 &T 同時存在
  • 在 activated 階段:&mut T 以完全獨占的方式使用
  • 某些情況指以下三種情況之一(上述鏈接對具體例子都有分析):
  • 調(diào)用 receiver 為 &mut self 的方法(包括方法調(diào)用時的自動引用):如 vec.push(vec.len())
  • 函數(shù)參數(shù)中的 &mut T reborrow:如 std::mem::replace(r, vec![r.len()])
  • 重載的復(fù)合賦值運算符中隱式的 &mut T:如本小節(jié)示例
  • 代碼中,任何顯式的 &mutref mut 都不是兩階段借用

MIR 可以幫助你看到兩階段借用。

bb0: { // reservation 階段
    _3 = &mut _1; // 兩階段借用的第三種前提:重載的復(fù)合賦值運算符中隱式的 `&mut T`
    _5 = &_1;     // `&mut T` 暫時被視為 `&T`,從而允許在此處使用 `&T`
    _4 = S::no_op(move _5) -> bb1;
}
bb1: { // activated 階段
    _2 = <S as AddAssign<()>>::add_assign(move _3, move _4) -> bb2;
}

實戰(zhàn)

例子源自 #72199 issue,@steffahn 做了很好的 解釋,這里從 MIR 角度進行補充。

Vec<i32>v[i] += v[j]

fn main() {
    let mut v = Vec::from([0, 1]); // 為了讓 MIR 精簡,故意不使用 vec![0, 1]
    v[0] += v[1]; // 第一步:i32 += i32
}

// 兩側(cè)為 primitive types, RTL: <i32 as Add<i32>>::add_assign(&mut v[0], v[1])
// 1. 計算 v[1]:對它脫糖 `<Vec<i32> as Index<usize>>::index(&v, 1)` 得到 `&i32`,然后解引用得到 `i32` 
// 2. 計算 &mut v[0]:對它脫糖 `<Vec<i32> as IndexMut<usize>>::index_mut(&mut v, 0)` 得到 `&mut i32`
// 可以看到先使用了 `&v`,再使用了 `&mut v`,通過借用檢查

// 僅列出 MIR 中的重點
// let mut _1: std::vec::Vec<i32>;
// bb1: {
//     _5 = &_1;
//     _4 = <Vec<i32> as Index<usize>>::index(move _5, const 1_usize) -> [return: bb2, unwind: bb6];
// }
// bb2: {
//     _3 = (*_4);
//     _7 = &mut _1;
//     _6 = <Vec<i32> as IndexMut<usize>>::index_mut(move _7, const 0_usize) -> [return: bb3, unwind: bb6];
// }
// bb3: {
//     _8 = CheckedAdd((*_6), _3);
//     assert(!move (_8.1: bool), "attempt to compute `{} + {}`, which would overflow", (*_6), move _3) -> [success: bb4, unwind: bb6];
// }
// bb4: {
//     (*_6) = move (_8.0: i32);
//     drop(_1) -> bb5;
// }

&mut [Custom]v[i] += v[j]

#[derive(Clone, Copy)]
struct MyNum(i32);

impl std::ops::AddAssign for MyNum {
    fn add_assign(&mut self, rhs: MyNum) {
        *self = MyNum(self.0 + rhs.0)
    }
}

fn main() {
    let mut b = vec![MyNum(0), MyNum(1)];
    let v = b.as_mut_slice();
    v[0] += v[1]; // MyNum += MyNum
}

// LTR: <MyNum as Add<MyNum>>::add_assign(&mut v[0], v[1])
// 1. 計算 &mut v[0]:獲取和維持對第 0 元素的獨占引用,但只進入 reservation 階段,將 &mut 視為 &,從而繼續(xù)使用切片
// 2. 計算 v[1]:在 `&mut v[0]` 的第一階段,通過 `*_10` 和索引拷貝 MyNum
// 3. 調(diào)用方法,`&mut v[0]` 進入 activated 階段

// 僅列出 MIR 中的重點
// let mut _1: std::vec::Vec<MyNum>;
// bb2: {
//     _11 = &mut _1;
//     _10 = Vec::<MyNum>::as_mut_slice(move _11) -> [return: bb3, unwind: bb8];
// }
// bb3: { // 索引前進行了邊界檢查
//     _14 = const 0_usize;
//     _15 = Len((*_10));
//     _16 = Lt(_14, _15);
//     assert(move _16, "index out of bounds: the length is {} but the index is {}", move _15, _14) -> [success: bb4, unwind: bb8];
// }
// bb4: {
//     _13 = &mut (*_10)[_14]; // 獲取 &mut v[0],進入 reservation 階段
//     _18 = const 1_usize; // 索引前進行了邊界檢查
//     _19 = Len((*_10));
//     _20 = Lt(_18, _19);
//     assert(move _20, "index out of bounds: the length is {} but the index is {}", move _19, _18) -> [success: bb5, unwind: bb8];
// }
// bb5: {
//     _17 = (*_10)[_18]; // 計算 v[1]
//     _12 = <MyNum as AddAssign>::add_assign(move _13, move _17) -> [return: bb6, unwind: bb8]; // activated 階段
// }

Vec<Custom>v[i] += v[j]

#[derive(Clone, Copy)]
struct MyNum(i32);

impl std::ops::AddAssign for MyNum {
    fn add_assign(&mut self, rhs: MyNum) {
        *self = MyNum(self.0 + rhs.0)
    }
}

fn main() {
    let mut b = vec![MyNum(0), MyNum(1)];
    b[0] += b[1];
}

它無法編譯成功,但編譯器提示你怎么 解決(把右側(cè)的值賦給局部變量,然后使用該變量):

error[E0502]: cannot borrow `b` as immutable because it is also borrowed as mutable
  --> src/main.rs:12:13
   |
12 |     b[0] += b[1];
   |     --------^---
   |     |       |
   |     |       immutable borrow occurs here
   |     mutable borrow occurs here
   |     mutable borrow later used here
   |
help: try adding a local storing this...
  --> src/main.rs:12:13
   |
12 |     b[0] += b[1];
   |             ^^^^
help: ...and then using that local here
  --> src/main.rs:12:5
   |
12 |     b[0] += b[1];
   |     ^^^^^^^^^^^^

當(dāng)你試著從 MIR 分析為什么這樣,你會發(fā)現(xiàn) playground 因為編譯失敗而沒有 MIR 的結(jié)果,提示為
Unable to locate file for Rust MIR output

此時,你仍可以在本地獲取一部分 MIR 結(jié)果,因為 MIR 其實經(jīng)過許多次迭代,mir_dump 文件夾下保留了半成品:運行
cargo rustc -- -Z dump-mir=main,查看 mir_dump/simd.main.-------.renumber.0.mir 文件。

// 僅列出關(guān)鍵部分
bb4: {
    _13 = &mut _1;
    _12 = <Vec<MyNum> as IndexMut<usize>>::index_mut(move _13, const 0_usize) -> [return: bb5, unwind: bb9];
}
bb5: {
    _11 = &mut (*_12);
    StorageDead(_13);
    StorageLive(_14);
    StorageLive(_15);
    StorageLive(_16);
    _16 = &_1;
    _15 = <Vec<MyNum> as Index<usize>>::index(move _16, const 1_usize) -> [return: bb6, unwind: bb9];
}
bb6: {
    _14 = (*_15);
    StorageDead(_16);
    _10 = <MyNum as AddAssign>::add_assign(move _11, move _14) -> [return: bb7, unwind: bb9];
}

把它與上一小節(jié)在 &mut [MyNum] 的 MIR 進行對比,你會發(fā)現(xiàn)在 &mut Vec<MyNum> 上沒有發(fā)生兩階段借用:

  • 觀察兩個 MIR 片段的 move _13,第二個片段的 &mut _1 借用已經(jīng)在獲取索引時結(jié)束(未能到達 add_assign),而第一個在調(diào)用 add_assign 時結(jié)束
  • 所以 Vec<MyNum> 上的 b[0] += b[1] 是通過兩個不同的 &mut Vec<MyNum>&Vec<MyNum>,分別得到 &mut MyNumMyNum 兩個操作數(shù)

而 Rust 的借用檢查不允許在一個函數(shù)調(diào)用中對同一個值同時使用 &mut&,從而編譯報錯。

// b[0] += b[1] on &mut [MyNum]

_10 = Vec::<MyNum>::as_mut_slice(move _11) // _10: &mut [MyNum]

_13 = &mut (*_10)[_14]; // two-phase
_17 = (*_10)[_18];      // reservation 階段

_12 = <MyNum as AddAssign>::add_assign(move _13, move _17) // activated 階段

// b[0] += b[1] on Vec<MyNum>

_13 = &mut _1; // _1: Vec<MyNum>
_12 = <Vec<MyNum> as IndexMut<usize>>::index_mut(move _13, const 0_usize) // _13: &mut Vec<MyNum>, _12: &mut MyNum
_11 = &mut (*_12); // reborrow

_16 = &_1;
_15 = <Vec<MyNum> as Index<usize>>::index(move _16, const 1_usize) // _16: &Vec<MyNum>, _15: &mut MyNum
_14 = (*_15);

_10 = <MyNum as AddAssign>::add_assign(move _11, move _14)
  • 總結(jié) += 是可重載的復(fù)合賦值運算符,Self += Rhs 脫糖為 <Self as AddAssign<Rhs>>::add_assign(&mut Self, Rhs),但
  • 對兩側(cè)為 primitive types 的運算數(shù),先計算 Rhs,再計算 Self,然后調(diào)用編譯器實現(xiàn)的相加函數(shù)
  • 若至少有一側(cè)運算數(shù)不為 primitive types,則先計算 Self,再計算 Rhs,然后調(diào)用重載后的實現(xiàn)(即 <Self as AddAssign<Rhs>>::add_assign
  • 大多數(shù)表達式是從左到右執(zhí)行的。從右到左是特殊情況,比如
  • 賦值表達式中,先計算 = 右側(cè)的值,再計算左側(cè)
  • 復(fù)合賦值表達式中,兩側(cè)為 primitive types 的運算數(shù)時,先計算復(fù)合賦值運算符右側(cè),再計算左側(cè)
  • MIR 是 Rust 編譯過程的重要一環(huán),(無論在代碼編譯成功還是失敗的情況下)也可以成為輔助你分析的 Rust 代碼的工具
  1. 一個例子

  2. 另一個例子 

  3. 拓展閱讀:運用這套流程分析 == 操作符的具體的例子 

到此這篇關(guān)于Rust += 運算符與 MIR 應(yīng)用的文章就介紹到這了,更多相關(guān)Rust += 運算符內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:

相關(guān)文章

  • Rust聲明宏在不同K線bar類型中的應(yīng)用小結(jié)

    Rust聲明宏在不同K線bar類型中的應(yīng)用小結(jié)

    在K線bar中,往往有很多不同分時k線圖,比如1,2,3,5,,,,,60,120,250,300…,,不同分鐘類型,如果不用宏,那么手寫會比較麻煩,下面就試用一下宏來實現(xiàn)不同類型的bar,感興趣的朋友一起看看吧
    2024-05-05
  • rust中的match表達式使用詳解

    rust中的match表達式使用詳解

    在rust中提供了一個極為強大的控制流運算符match,這篇文章主要介紹了rust中的match表達式,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-08-08
  • Rust 語言中的 into() 方法及代碼實例

    Rust 語言中的 into() 方法及代碼實例

    在 Rust 中,into() 方法通常用于將一個類型的值轉(zhuǎn)換為另一個類型,這通常涉及到資源的所有權(quán)轉(zhuǎn)移,本文給大家介紹Rust 語言中的 into() 方法及代碼實例,感謝的朋友跟隨小編一起看看吧
    2024-03-03
  • 詳解Rust語言中anyhow的使用

    詳解Rust語言中anyhow的使用

    anyhow是一個Rust庫,用于簡化錯誤處理和提供更好的錯誤報告,這個庫適合用于應(yīng)用程序,而不是用于創(chuàng)建庫,因為它提供了一個非結(jié)構(gòu)化的,方便使用的錯誤類型,本文就給大家講講Rust語言中anyhow的使用,需要的朋友可以參考下
    2023-08-08
  • 一文詳解Rust中的錯誤處理

    一文詳解Rust中的錯誤處理

    這篇文章主要為大家詳細介紹了Rust中的錯誤處理的相關(guān)知識,文中的示例代碼講解詳細,具有一定的借鑒價值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-01-01
  • Rust 標(biāo)準庫的結(jié)構(gòu)及模塊路徑詳解

    Rust 標(biāo)準庫的結(jié)構(gòu)及模塊路徑詳解

    在 Rust 中,標(biāo)準庫提供了一組核心功能,以幫助開發(fā)者執(zhí)行常見的編程任務(wù),這個路徑樹可以作為參考,幫助你更好地理解 Rust 標(biāo)準庫的結(jié)構(gòu)和模塊之間的關(guān)系,本文介紹 Rust 標(biāo)準庫的結(jié)構(gòu),并提供相應(yīng)的 use 路徑,感興趣的朋友一起看看吧
    2024-05-05
  • 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多個mod文件引用和文件夾mod使用注意事項小結(jié)

    rust多個mod文件引用和文件夾mod使用注意事項小結(jié)

    在 Rust 項目中,可以使用 mod 關(guān)鍵字將一個文件夾或一個 rs 文件作為一個模塊引入到當(dāng)前文件中,本文給大家介紹rust多個mod文件引用和文件夾mod使用注意事項小結(jié),感興趣的朋友跟隨小編一起看看吧
    2024-03-03
  • rust使用Atomic創(chuàng)建全局變量和使用操作方法

    rust使用Atomic創(chuàng)建全局變量和使用操作方法

    從 Rust1.34 版本后,就正式支持原子類型,原子指的是一系列不可被 CPU 上下文交換的機器指令,這些指令組合在一起就形成了原子操作,這篇文章主要介紹了rust使用Atomic創(chuàng)建全局變量和使用,需要的朋友可以參考下
    2024-05-05
  • Rust?Atomics?and?Locks?源碼解讀

    Rust?Atomics?and?Locks?源碼解讀

    這篇文章主要為大家介紹了Rust?Atomics?and?Locks?源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-02-02

最新評論