C++中 ‘=default ’及‘ =delete ’的使用

前言:

C++的類有四類特殊成員函數,它們分別是:默認構造函數、析構函數、拷貝構造函數、拷貝賦值運算符。如果實際編碼時沒有顯示定義,那麼編譯器將會默認生成這四類成員函數。使用=default=delete可以控制編譯器默認函數體的使用。

1 =default

C++11新增瞭=default標識,編譯器看到後,會生成默認的執行效率更高的函數定義體,同時會減輕編碼時的工作量。當然,這裡會引入一個問題,既然編譯器會默認生成構造函數,那麼=default的優勢在哪裡呢?回答這個問題之前,先看下這段代碼

class Test{
public:
    Test(int a):x(a){};
private:
    int x;
};
int main ()
{
  Test test;
  return 0;
}


眾所周知,上面這段是編譯不過的,原因是因為在Test類中我們自己定義瞭一個構造函數,編譯器看到後就不會再生成默認構造函數給我們,如果要解決這個編譯問題的話需要我們提供一個沒有參數的構造函數。

如:

 Test(){};


在類中加瞭上面的代碼之後,編譯器就會編譯通過,但是在試想一下,如果這個類很大,且需要我們在類中初始化的成員很多呢?這個時候我們需要提供的這個默認構造函數就變得的很龐大,浪費我們很多時間進行變量的初始化,寫一堆沒有技術的賦值或者其它初始化語句。同樣,拷貝構造函數和拷貝賦值函數也是一樣。

=default就給我們提供瞭這樣一個功能,加上之後,編譯器就會給我們默認生成函數體,減輕工作量。

上面的類就這可以這些寫:

class Test{
public:
    Test(int a):x(a){};
    Test()=default;
private:
    int x;
};


當然,=default不但可以在類成員內部添加也可以在類之外添加,但是使用=default時,必須遵守一個準則:default 函數特性隻能用於類的特殊成員函數或者函數沒有默認參數。 =default

寫在類之外的方式如下:

class Test{
public:
    Test(int a):x(a){};
    Test()=default;
    Test(const Test& ts);
    Test& operator = (const Test& ts);
private:
    int x;
};
Test::Test(const Test& ts) =default;
Test& Test::operator = (const Test& ts) =default;


上面的代碼中演示瞭=default在類成員外部使用的場景。但是類中確沒有析構函數,編碼時,如果涉及到類的繼承和派生,尤其是通過基類指針指向瞭派生類對象,當調用delete刪除派生對象時,如果基類沒有顯示定義析構函數,編譯器會為基類默認生成析構函數,基類對象會被正常釋放,但是也會產生一個問題,派生類沒有正確釋放,可能會產生內存泄露等問題。正確解決這種問題的做法是在基類中顯示定義一個虛析構函數。這種方法在C++11之前是我們解決這種問題經常使用的,但是C++11之後,可以使用=default ,從而減輕我們的編碼量,且編譯器生成的代碼效率更高。

代碼如下所示:

class Base{
public:
    virtual ~Base()=default;
private:
    int x;
};
class A : public Base{
private:
    int y;
};
int main ()
{
  Base *pBase = new A;
  delete pBase;
  return 0;
}


2 =delete

C++11之前,delete是和new配對使用的,釋放程序在堆上開辟得空間,將資源返還給操作系統,C++11之後,delete又多瞭一個含義既:禁用成員函數的使用。使用方法為:在函數名稱後面加上=delete。

下面代碼定義瞭一個類,類裡面定義瞭一個整型的成員變量,在main函數中使用時,創建瞭兩個類的實例,一個傳入參數用整型一個用浮點型,

代碼如下:

class Test { 
public: 
    Test()=delete;
    Test(int a):x(a) {std::cout<<x<<std::endl;} 
private:
    int x;
}; 
int main() { 
    Test test1(1); 
    Test test2(1.1);  
    return 0; 
} 


如上,這段代碼是可以編譯通過的,因為這段代碼在編譯時發生瞭隱式轉換,將浮點型數據轉成瞭整型,代價是損失瞭精度。代碼運行後輸出的結果都是:1。

如果不想在傳入非整型的數據時編譯通過,就可以使用=delete來抑制這種問題的產生。如使用=delete解決上面的問題,

代碼如下:

class Test { 
public: 
    Test(int a):x(a) {std::cout<<x<<std::endl;} 
    Test(double)=delete;
private:
    int x;
}; 
int main() { 
    Test test1(1); 
    Test test2(1.1);  
    return 0;


編譯時報錯信息為:

main.cpp:23:19: error: use of deleted function ‘Test::Test(double)'
   23 |     Test test2(1.1);
      |                   ^
main.cpp:17:5: note: declared here
   17 |     Test(double)=delete;
      |     ^~~~


如上可知,使用=delete後,可以使我們禁用一些不需要編譯器生成的默認函數,還可以避免因為數據類型原因導致的錯誤的函數調用。

到此這篇關於C++中 =default 及 =delete 使用的文章就介紹到這瞭,更多相關C++中 =default 和 =delete 使用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: