通過學習C++11移動建構式我們知道,C++11 標準中借助右值參考可以為指定類添加移動建構式,這樣當使用該類的右值物件(可以理解為臨時物件)初始化同類物件時,編譯器會優先選擇移動建構式,
需要注意,移動建構式的呼叫時機是:用同類的右值物件初始化新物件,那么,用當前類的左值物件(有名稱,能獲取其存盤地址的實體物件)初始化同類物件時,是否就無法呼叫移動建構式了呢?當然不是,C++11 標準中已經給出了解決方案,即呼叫 move() 函式,
move 本意為 "移動",但該函式并不能移動任何資料,它的功能很簡單,就是將某個左值強制轉化為右值,基于 move() 函式特殊的功能,其常用于實作移動語意,
move() 函式的用法也很簡單,其語法格式如下:
move( arg )
其中,arg 表示指定的左值物件,該函式會回傳 arg 物件的右值形式,
下面用一些代碼演示了move() 函式的使用方法,
(1)move() 函式的基礎應用:
#include <iostream>
using namespace std;
class movedemo{
public:
movedemo():num(new int(0)){
cout<<"construct!"<<endl;
}
//拷貝建構式
movedemo(const movedemo &d):num(new int(*d.num)){
cout<<"copy construct!"<<endl;
}
//移動建構式
movedemo(movedemo &&d):num(d.num){
d.num = NULL;
cout<<"move construct!"<<endl;
}
public: //這里應該是 private,使用 public 是為了更方便說明問題
int *num;
};
int main(){
movedemo demo;
cout << "demo2:\n";
movedemo demo2 = demo;
//cout << *demo2.num << endl; //可以執行
cout << "demo3:\n";
movedemo demo3 = std::move(demo);
//此時 demo.num = NULL,因此下面代碼會報運行時錯誤
//cout << *demo.num << endl;
return 0;
}
程式執行結果為:
construct!
demo2:
copy construct!
demo3:
move construct!
通過觀察程式的輸出結果,以及對比 demo2 和 demo3 初始化操作不難得知,demo 物件作為左值,直接用于初始化 demo2 物件,其底層呼叫的是拷貝建構式;而通過呼叫 move() 函式可以得到 demo 物件的右值形式,用其初始化 demo3 物件,編譯器會優先呼叫移動建構式,
注意:呼叫拷貝建構式,并不影響 demo 物件,但如果呼叫移動建構式,由于函式內部會重置 demo.num 指標的指向為 NULL,所以程式中第 30 行代碼會導致程式運行時發生錯誤,
(2)靈活使用 move() 函式:
#include <iostream>
using namespace std;
class first {
public:
first() :num(new int(0)) {
cout << "construct!" << endl;
}
//移動建構式
first(first &&d) :num(d.num) {
d.num = NULL;
cout << "first move construct!" << endl;
}
public: //這里應該是 private,使用 public 是為了更方便說明問題
int *num;
};
class second {
public:
second() :fir() {}
//用 first 類的移動建構式初始化 fir
second(second && sec) :fir(move(sec.fir)) {
cout << "second move construct" << endl;
}
public: //這里也應該是 private,使用 public 是為了更方便說明問題
first fir;
};
int main() {
second oth;
second oth2 = move(oth);
//cout << *oth.fir.num << endl; //程式報運行時錯誤
return 0;
}
程式執行結果為:
construct!
first move construct!
second move construct
程式中分別構建了 first 和 second 這 2 個類,其中 second 類中包含一個 first 類物件,仔細觀察可以發現,程式中使用了兩次 move() 函式:
- 程式第 31 行:由于 oth 為左值,如果想呼叫移動建構式為 oth2 初始化,需先利用 move() 函式生成一個 oth 的右值版本;
- 程式第 22 行:oth 物件內部還包含一個 first 類物件,對于 oth.fir 來說,其也是一個左值,所以在初始化 oth.fir 時,還需要再呼叫一次 move() 函式,
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/541181.html
標籤:其他
上一篇:Spring Boot常見問題
下一篇:前端菜鳥如何快速開發個人產品
