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

C++移動語義詳細介紹使用

 更新時間:2023年01月27日 09:41:40   作者:Shawn-Summer  
首先,移動語義和完美轉(zhuǎn)發(fā)這兩個概念是在C++的模板編程的基礎(chǔ)上,新增的特性,主要是配合模板來使用。本篇會從C++的值類型,到移動拷貝與移動賦值來理解移動語義與完美轉(zhuǎn)發(fā)

1.移動構(gòu)造函數(shù)

移動語義就是使用移動構(gòu)造函數(shù)來構(gòu)造對象。

我們知道在類中如果存在指針數(shù)據(jù)成員,那么我們就一定要寫拷貝構(gòu)造函數(shù),進行深拷貝

如下所示,就是拷貝構(gòu)造函數(shù)的用法:

#include<iostream>
using namespace std;
class A
{
    public:
        int * ptr;
        A():ptr(new int(0))
        {
            cout<<"constructor\n";
        };
        A(const A & h):ptr(new int(*h.ptr))//拷貝構(gòu)造函數(shù)
        {
            cout<<"copy constructor\n";
            //deep copy
        };
        ~A()
        {
            cout<<"destructor\n";
            delete ptr;
        };
};
A getA()
{
    return A();
}
int main()
{
    A a=getA();
}
//g++ .\test.cpp -std=c++11 -fno-elide-constructors

constructor
copy constructor
destructor
copy constructor
destructor
destructor

可以知道上面代碼中,實際上產(chǎn)生了3個對象,在getA()函數(shù)中,使用默認構(gòu)造函數(shù)產(chǎn)生一個對象,然后將其作為返回值時,又會通過拷貝構(gòu)造函數(shù)產(chǎn)生一個對象,然后在main()函數(shù)中,又會通過拷貝構(gòu)造函數(shù)構(gòu)造出對象a,所以總共有3個對象產(chǎn)生,我們這里的拷貝構(gòu)造函數(shù)是進行的深拷貝,所以就會開辟3塊內(nèi)存.

在C++11中,我們可以使用移動構(gòu)造函數(shù),對上述代碼進行優(yōu)化

#include<iostream>
using namespace std;
class A
{
    public:
        int * ptr;
        A():ptr(new int(0))
        {
            cout<<"constructor\n";
        };
        A(const A & h):ptr(new int(*h.ptr))//拷貝構(gòu)造函數(shù)
        {
            cout<<"copy constructor\n";
            //deep copy
        };
        A(A && h):ptr(h.ptr)//移動構(gòu)造函數(shù)
        {
            h.ptr=nullptr;
            cout<<"move constructor\n";
        }
        ~A()
        {
            cout<<"destructor\n";
            delete ptr;
        };
};
A getA()
{
    return A();
}
int main()
{
    A a=getA();
}
//g++ .\test.cpp -std=c++11 -fno-elide-constructors

constructor
move constructor
destructor
move constructor
destructor
destructor

移動構(gòu)造函數(shù),它是進行的淺拷貝,由于被移動的值會立即進行析構(gòu),所以我們不關(guān)心它,只需進行淺拷貝,將其開辟的內(nèi)存空間轉(zhuǎn)讓給別人。上述代碼中,也會構(gòu)造出3個對象,但是它們只開辟一塊內(nèi)存空間,這就是移動構(gòu)造函數(shù)的優(yōu)勢。

總之,我們發(fā)現(xiàn)移動構(gòu)造函數(shù)和拷貝構(gòu)造函數(shù)的區(qū)別,其實就是深拷貝和淺拷貝的區(qū)別,移動構(gòu)造函數(shù)的開銷更小,當然我們關(guān)心的是,移動構(gòu)造函數(shù)何時會被觸發(fā)?在上面代碼中就是一個例子,將getA()中的局部匿名對象移動給返回值,然后將返回值移動給 main()中的a。

這里我們給出結(jié)論:移動構(gòu)造函數(shù)只有在使用右值或右值引用來構(gòu)造對象時才會調(diào)用

那么什么是右值?

getA()中的A()就是右值,getA()的返回值也是右值,所以用它們構(gòu)造對象時,會調(diào)用移動構(gòu)造函數(shù)

那么什么是右值引用?

顧名思義就是右值的引用

2.右值引用

在C++11中,我們將值劃分為:左值、右值(分為將亡值和純右值)

左值:可以取地址,有名字的值

右值:不能取地址,沒有名字的值

純右值:運算表達式,如1+2,或者和對象無關(guān)的字面值,如true,或者非引用的函數(shù)返回值,或者lambda表達式

將亡值:僅和右值引用相關(guān)的值,它包括:右值引用的函數(shù)返回值T&&,或者std::move的返回值,或者被轉(zhuǎn)換為T&&類型的函數(shù)返回值

注意:不管是純右值還是將亡值,它們的存活時間都很短。不要被將亡值的名稱所迷惑了,其實所以右值的都會即將消亡。

實際上,對于純右值和將亡值的定義很難給出,而且我們也不需要區(qū)分它們兩,但是,我們至少可以確定一個值是左值還是右值。

C++98中所提及的引用,在C++11中我們稱之為左值引用,即這個引用只能綁定左值,在C++11中我們提供了一種新的能夠綁定右值的引用,即右值引用。

我們知道左值引用實際是一個變量的別名,右值引用它實際是一個匿名變量的別名

#include<iostream>
using namespace std;
class A
{
    public:
        int * ptr;
        A():ptr(new int(0))
        {
            cout<<"constructor\n";
        };
        A(const A & h):ptr(new int(*h.ptr))//拷貝構(gòu)造函數(shù)
        {
            cout<<"copy constructor\n";
            //deep copy
        };
        A(A && h):ptr(h.ptr)//移動構(gòu)造函數(shù)
        {
            h.ptr=nullptr;
            cout<<"move constructor\n";
        }
        ~A()
        {
            cout<<"destructor\n";
            delete ptr;
        };
};
A getA()
{
    return A();
}
int main()
{
    A&& a=getA();//右值引用
}

constructor
move constructor
destructor
destructor

在上述代碼中getA()的返回值是一個右值,它是一個臨時值,如果我們寫成A a=getA();,那么這個臨時值給a進行移動構(gòu)造后就會立即被析構(gòu),而如果我們使用A&& a=getA();,那就意味著我們給這個臨時值進行續(xù)命,a就是這個臨時值的別名,所以上述代碼就會少一個對象的構(gòu)造。

總之,右值引用就是一種綁定右值的引用,實際上在C++98中,我們所知的const T &,這樣的引用,也可以綁定右值,他也叫做萬能引用,當他綁定右值的時候它的作用和右值引用是一樣的,只不過這里的const是底層的,所以我們不能用其修改右值,所以右值引用綁定右值時,可以修改該右值,而當萬能引用綁定右值時,我們不可以修改該右值

T& a;//左值引用,只能綁定非常量左值
T&& a;//右值引用,只能綁定非常量右值
const T& a;//萬能引用,它可以綁定一切值,但是它不能修改該值
const T&& a;//和萬能引用功能一樣(一般不使用)

我們仔細來思索一下右值引用的用處,從本質(zhì)上講,它是給右值進行續(xù)命,而從實踐上講,它就是用來移動語義的,但是移動語義的時候,我們希望修改原來的右值(看上面代碼中的移動構(gòu)造函數(shù),它實際上修改了右值),所以我們說const T&&這種是無用的,

我們在學習了C++11中的移動語義和右值引用知識后,我們要深知一個編程規(guī)矩:

只要類中有指針數(shù)據(jù)成員,就一定要重寫拷貝構(gòu)造函數(shù)和移動構(gòu)造函數(shù)

3.std::move()將左值強制轉(zhuǎn)換為右值引用

看一下下面這段代碼

#include<iostream>
using namespace std;
class A
{
    public:
        int * ptr;
        A():ptr(new int(0))
        {
            cout<<"constructor\n";
        };
        A(const A & h):ptr(new int(*h.ptr))//拷貝構(gòu)造函數(shù)
        {
            cout<<"copy constructor\n";
            //deep copy
        };
        A(A && h):ptr(h.ptr)//移動構(gòu)造函數(shù)
        {
            h.ptr=nullptr;
            cout<<"move constructor\n";
        }
        ~A()
        {
            cout<<"destructor\n";
            delete ptr;
        };
};
A&& getA()
{
    return std::move(A());
}
int main()
{
    A&& a=getA();
}

constructor
destructor

getA()中的A()是右值,為什么還要用std::move將其轉(zhuǎn)換為右值引用?因為A()是一個純右值,右值引用當然可以綁定純右值,但是A()是一個局部對象,在函數(shù)中返回引用時,我們禁止返回局部對象的引用,但是當我們使用std::move后,A()就會轉(zhuǎn)換為右值引用類型,這樣子就可以將其作為引用返回。這是一種返回局部對象引用的特殊方法。

注意,這是一個涉及原則的問題,匿名對象是純右值

class  A
{};
int main()
{
    A& a=A();//報錯,左值引用無法綁定純右值
}

實際上,std::move()等價于static_cast<T&&>(lvalue),即將左值轉(zhuǎn)換為右值引用。

但是,std::move()有一個bug,即被轉(zhuǎn)化為右值引用的左值,不會被立即析構(gòu)。

#include<iostream>
using namespace std;
class A
{
    public:
        int* ptr;
        A():ptr(new int(999)){}
        ~A(){delete ptr;}
        A(const A& h):ptr(new int(*h.ptr)){}
        A(A&& h):ptr(h.ptr)
        {
            h.ptr=nullptr;
        }
};
int main()
{
    A a;
    A b(std::move(a));
    cout<<*a.ptr<<endl;//報錯
}

上述代碼就會報錯,因為a被轉(zhuǎn)化為右值引用后,b會調(diào)用移動構(gòu)造函數(shù)來構(gòu)造它自己,而在移動構(gòu)造函數(shù)中,它將a.ptr置空

#include<utility>
class A
{
    public:
        int *ptr;
        A():ptr(new int(0)){}
        ~A(){delete ptr;}
        A(const A& h):ptr(new int(*h.ptr)){}
        A(A&& h):ptr(h.ptr){h.ptr=nullptr;}
};
class B
{
    public:
        int *ptr;
        A elem;
        B():ptr(new int(0)){}
        ~B(){delete ptr;}
        B(const B&h):ptr(new int(*h.ptr)),elem(h.elem){}
        B(B&& h):ptr(h.ptr),elem(std::move(h.elem)){h.ptr=nullptr;}
};

注意看,B(const B&h):ptr(new int(*h.ptr)),elem(h.elem){}中對elem的初始化使用的是A的拷貝構(gòu)造函數(shù),

B(B&& h):ptr(h.ptr),elem(std::move(h.elem)){h.ptr=nullptr;}中對elem的初始化使用的是是A的移動構(gòu)造函數(shù). 注意一點,即使這里我們忘記寫std::move()也并無大礙,它會自行調(diào)用拷貝構(gòu)造函數(shù),當然這也會導致一些開銷,所以在做類開發(fā)的時候,在寫類的移動構(gòu)造函數(shù)的時候,總是要記得將類成員move成右值引用。

4.拷貝語義和移動語義

如果一個類支持拷貝構(gòu)造函數(shù)和拷貝賦值函數(shù),那么我們就稱該類具有拷貝語義;同樣的如果一個類支持移動構(gòu)造函數(shù)和移動賦值函數(shù),那么我們就稱該類具有移動語義。

當然有些類是同時支持移動語義和拷貝語義的。

在C++98中的類基本都是只具有拷貝語義的,而在C++11中的基本所有類都支持移動語義,特別的,有些類只支持移動語義,而不支持拷貝語義,這種類,我們稱之為資源型類,即資源只能被移動而不能被拷貝,例如智能指針類unique_ptr,文件流ifstream等都是資源型類,在C++11中,我們可以通過一些工具來判斷一個類是否支持移動語義。

我們看一下下面的代碼

template <class T>
void swap(T& a, T& b)
{
    T tmp(move(a));
    a=move(b);
    b=move(tmp);
}

上述代碼中,如果T支持移動語義,那么它就會調(diào)用移動構(gòu)造函數(shù)和移動賦值函數(shù),而如果T只支持拷貝語義,那么它也可以調(diào)用拷貝構(gòu)造函數(shù)和拷貝賦值函數(shù)

我們關(guān)于移動語義的另一個話題是:異常。因為如果移動語義沒有完成,卻拋出異常,那么可能會導致產(chǎn)生懸掛指針。所以在C++11中我們同樣有std::move_if_noexcept()函數(shù)來檢測,移動構(gòu)造函數(shù)是否用noexcept修飾。

再討論一個關(guān)于編譯器優(yōu)化的問題,如今c++編譯器已經(jīng)非常優(yōu)化了,RVO機制,即所謂返回值優(yōu)化機制,他能幫你完成類似移動語義的智能優(yōu)化,但是要記住,編譯器優(yōu)化不是完全奏效的,最好還是自己提高代碼效率。

到此這篇關(guān)于C++移動語義詳細介紹使用的文章就介紹到這了,更多相關(guān)C++移動語義內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用C語言實現(xiàn)珠璣妙算Mastermind小游戲

    使用C語言實現(xiàn)珠璣妙算Mastermind小游戲

    這篇文章主要介紹了使用C語言實現(xiàn)珠璣妙算Mastermind小游戲,這是一款益智類多人游戲游戲,非常有趣,需要的朋友可以參考下
    2023-03-03
  • C語言中#define預處理語法總結(jié)

    C語言中#define預處理語法總結(jié)

    C語言里可以用#define定義一個標識符來表示一個常量。特點是:定義的標識符不占內(nèi)存,只是一個臨時的符號,預編譯后這個符號就不存在了,也不做類型定義。預編譯又叫預處理
    2021-11-11
  • c語言swap(a,b)值交換的4種實現(xiàn)方法

    c語言swap(a,b)值交換的4種實現(xiàn)方法

    c語言swap(a,b)值交換的4種實現(xiàn)方法,這么好的東西,盡管簡單,但值得發(fā)表,以此共享。
    2013-02-02
  • 非常經(jīng)典的C語言趣味題目

    非常經(jīng)典的C語言趣味題目

    在這個網(wǎng)站上發(fā)現(xiàn)一套很有趣的C語言測試題,如果你招聘C語言相關(guān)開發(fā)人員,或者正在學習C語言,很值得做一做
    2013-04-04
  • C++ continue和break語句

    C++ continue和break語句

    這篇文章主要介紹了C++ continue和break語句,文章圍繞continue和break語句的相關(guān)資料展開詳細內(nèi)容,需要的朋友可以參考一下,希望對大家有所幫助
    2021-11-11
  • C++ OpenCV實戰(zhàn)之形狀識別

    C++ OpenCV實戰(zhàn)之形狀識別

    本案例通過使用OpenCV中的approxPolyDP進行多邊形近似,進而進行基礎(chǔ)形狀識別(圓、三角形、矩形、星形…),快跟隨小編一起動手嘗試一下
    2022-07-07
  • C++ 如何實現(xiàn)一個日期類

    C++ 如何實現(xiàn)一個日期類

    通過對類和對象的學習,理解了類是對象的抽象描述,實現(xiàn)日期類涉及定義年月日屬性及成員函數(shù)如打印日期、日期加減,重點介紹了運算符重載的概念和作用,通過代碼示例展示了如何實現(xiàn)一個日期類,包括頭文件和源文件的分離編寫
    2024-10-10
  • C語言?使用qsort函數(shù)來進行快速排序

    C語言?使用qsort函數(shù)來進行快速排序

    排序方法有很多種:選擇排序,冒泡排序,歸并排序,快速排序等。?看名字都知道快速排序是目前公認的一種比較好的排序算法。因為他速度很快,所以系統(tǒng)也在庫里實現(xiàn)這個算法,便于我們的使用。?這就是qsort函數(shù)
    2022-02-02
  • C++學習之線程詳解

    C++學習之線程詳解

    多線程是開發(fā)中必不可少的,往往我們需要多個任務(wù)并行,就需要多線程開發(fā)。本文將帶大家深入學習一下C++中的常用的一些知識點,感興趣的同學可以了解一下
    2021-12-12
  • C++ 成員變量的初始化順序問題詳解

    C++ 成員變量的初始化順序問題詳解

    這篇文章主要介紹了C++ 成員變量的初始化順序問題詳解的相關(guān)資料,需要的朋友可以參考下
    2017-02-02

最新評論