c++禁止函數(shù)的傳值調(diào)用的方法
代碼編譯運(yùn)行環(huán)境:VS2017+Debug+Win32
按照參數(shù)形式的不同,C++應(yīng)該有三種函數(shù)調(diào)用方式:傳值調(diào)用、引用調(diào)用和指針調(diào)用。對(duì)于基本數(shù)據(jù)類型的變量作為實(shí)參進(jìn)行參數(shù)傳遞時(shí),采用傳值調(diào)用與引用調(diào)用和指針調(diào)用的效率相差不大。但是,對(duì)于類類型來(lái)說(shuō),傳值調(diào)用和引用調(diào)用之間的區(qū)別很大,類對(duì)象的尺寸越大,這種差別越大。
傳值調(diào)用與后面兩者的區(qū)別在于傳值調(diào)用在進(jìn)入函數(shù)體之前,會(huì)在棧上建立一個(gè)實(shí)參的副本,而引用和指針調(diào)用沒(méi)有這個(gè)動(dòng)作。建立副本的操作是利用拷貝構(gòu)造函數(shù)進(jìn)行的。因此,要禁止傳值調(diào)用,就必須在類的拷貝構(gòu)造函數(shù)上做文章。
可以直接在拷貝構(gòu)造函數(shù)中拋出異常,這樣就迫使程序員不能使用拷貝構(gòu)造函數(shù),否則程序總是出現(xiàn)運(yùn)行時(shí)錯(cuò)誤。但是,這不是一個(gè)好的辦法,應(yīng)該在編譯的階段就告訴程序員,不能使用該類的拷貝構(gòu)造函數(shù)。
1.不顯示定義拷貝構(gòu)造函數(shù)可行嗎?
#include <iostream>
using namespace std;
class A
{
public:
int num;
A(){num=5;}
};
void show(A a)
{
cout<<a.num<<endl;
}
int main()
{
A obj;
show(obj);
}
以上程序順利通過(guò)編譯,并輸出5。因此,不顯示定義拷貝構(gòu)造函數(shù),并不能阻止對(duì)類的拷貝構(gòu)造函數(shù)的調(diào)用,原因是編譯器會(huì)自動(dòng)為沒(méi)有顯示定義拷貝構(gòu)造函數(shù)的類提供一個(gè)默認(rèn)的拷貝構(gòu)造函數(shù)。
2.顯示定義拷貝構(gòu)造函數(shù)并將訪問(wèn)權(quán)限設(shè)置為private
上面的程序添加拷貝構(gòu)造函數(shù)的定義,修改如下。
#include <iostream>
using namespace std;
class A
{
A(const A&){};
public:
int num;
A(){num=5;}
};
void show(A a)
{
cout<<a.num<<endl;
}
int main()
{
A obj;
show(obj);
}
這個(gè)程序在VS2017環(huán)境下編譯不通過(guò),得到如下錯(cuò)誤:error C2248: “A::A”: 無(wú)法訪問(wèn) private 成員(在“A”類中聲明)。
這樣就能阻止了函數(shù)調(diào)用時(shí),類A的對(duì)象以值傳遞的方式進(jìn)行函數(shù)函數(shù)調(diào)用。為使程序通過(guò)編譯,需將show()函數(shù)的定義改為如下形式:
void show(const A& a)
{
cout<<a.num<<endl;
}
3.拷貝構(gòu)造函數(shù)的說(shuō)明
(1)如果將拷貝構(gòu)造函數(shù)中的引用符號(hào)去掉&,編譯將無(wú)法通過(guò),出錯(cuò)的信息如下:非法的復(fù)制構(gòu)造函數(shù): 第一個(gè)參數(shù)不應(yīng)是“A”。原因是如果拷貝構(gòu)造函數(shù)中的參數(shù)不是一個(gè)引用,即形如A(const A a),那么就相當(dāng)于采用了傳值的方式(pass-by-value),而傳值的方式會(huì)調(diào)用該類的拷貝構(gòu)造函數(shù),從而造成無(wú)窮遞歸地調(diào)用拷貝構(gòu)造函數(shù)。因此拷貝構(gòu)造函數(shù)的參數(shù)必須是一個(gè)引用或一個(gè)指針。
(2)拷貝構(gòu)造函數(shù)的參數(shù)通常情況下是const的,但是const并不是嚴(yán)格必須的。
(3)附帶說(shuō)明,在下面幾種情況下會(huì)調(diào)用拷貝構(gòu)造函數(shù):
a. 顯式或隱式地用同類型的一個(gè)對(duì)象來(lái)初始化另外一個(gè)對(duì)象;
b. 作為實(shí)參以值傳遞的方式傳遞給一個(gè)函數(shù);
c. 在函數(shù)體內(nèi)返回一個(gè)對(duì)象時(shí),也會(huì)調(diào)用返回值類型的拷貝構(gòu)造函數(shù);
d. 需要產(chǎn)生一個(gè)臨時(shí)類對(duì)象時(shí)(類對(duì)象作為函數(shù)返回值會(huì)創(chuàng)建臨時(shí)對(duì)象)。
以上就是c++禁止函數(shù)的傳值調(diào)用的方法的詳細(xì)內(nèi)容,更多關(guān)于c++禁止函數(shù)的傳值調(diào)用的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
stl容器set,map,vector之erase用法與返回值詳細(xì)解析
在使用 list、set 或 map遍歷刪除某些元素時(shí)可以這樣使用,如下所示2013-09-09
C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單掃雷游戲
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-02-02
C語(yǔ)言 超詳細(xì)介紹與實(shí)現(xiàn)線性表中的帶頭雙向循環(huán)鏈表
帶頭雙向循環(huán)鏈表:結(jié)構(gòu)最復(fù)雜,一般用在單獨(dú)存儲(chǔ)數(shù)據(jù)。實(shí)際中使用的鏈表數(shù)據(jù)結(jié)構(gòu),都是帶頭雙向循環(huán)鏈表。另外這個(gè)結(jié)構(gòu)雖然結(jié)構(gòu)復(fù)雜,但是使用代碼實(shí)現(xiàn)以后會(huì)發(fā)現(xiàn)結(jié)構(gòu)會(huì)帶來(lái)很多優(yōu)勢(shì),實(shí)現(xiàn)反而簡(jiǎn)單2022-03-03
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)旋轉(zhuǎn)鏈表的實(shí)現(xiàn)
這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)旋轉(zhuǎn)鏈表的實(shí)現(xiàn)的相關(guān)資料,這里提供實(shí)例幫助大家實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-08-08

