C++缺省參數(shù)的具體使用
一、缺省參數(shù)概念
缺省參數(shù)是聲明或定義函數(shù)時(shí)為函數(shù)的參數(shù)指定一個(gè)默認(rèn)值。在調(diào)用該函數(shù)時(shí),如果沒有指定實(shí)參則采用該默認(rèn)值,否則使用指定的實(shí)參
#include<iostream>
using namespace std;
void TestFunc(int a = 0)//參數(shù)缺省值
{
?? ?cout << a << endl;
}
int main()
{
?? ?TestFunc();//沒有指定實(shí)參,使用缺省值
?? ?TestFunc(10);//指定實(shí)參,使用實(shí)參
?? ?return 0;
}? 有什么用 ?
比如在 C 語言中有個(gè)很苦惱的問題是寫棧時(shí),不知道要開多大的空間,之前我們是如果棧為空就先開 4 塊空間,之后再以 2 倍走,如果我們明確知道要很大的空間,那么這樣就只能一點(diǎn)一點(diǎn)的接近這塊空間,就太 low 了。但如果我們使用缺省,明確知道不需要太大時(shí)就使用默認(rèn)的空間大小,明確知道要很大時(shí)再傳參
#include<iostream>
using namespace std;
namespace WD
{
?? ?struct Stack
?? ?{
?? ??? ?int* a;
?? ??? ?int size;
?? ??? ?int capacity;?? ?
?? ?};
}
using namespace WD;
void StackInit(struct Stack* ps)
{
?? ?ps->a = NULL;?
?? ?ps->capacity = 0;
?? ?ps->size = 0;
}
void StackPush(struct Stack* ps, int x)
{
?? ?if(ps->size == ps->capacity)
?? ?{
?? ??? ?//ps->capacity *= 2;//err
?? ??? ?ps->capacity == 0 ? 4 : ps->capacity * 2;//這里就必須寫一個(gè)三目
?? ?}
}
void StackInitCpp1(struct Stack* ps, int defaultCP)
{
?? ?ps->a = (int*)malloc(sizeof(int) * defaultCP);
?? ?ps->capacity = 0;
?? ?ps->size = defaultCP;
}
void StackInitCpp2(struct Stack* ps, int defaultCP = 4)//ok
{
?? ?ps->a = (int*)malloc(sizeof(int) * defaultCP);
?? ?ps->capacity = 0;
?? ?ps->size = defaultCP;
}
int main()
{
?? ?//假設(shè)明確知道這里至少需要100個(gè)數(shù)據(jù)到st1
?? ?struct Stack st1;?
?? ?StackInitCpp1(&st1, 100);
?? ?//假設(shè)不知道st2里需要多少個(gè)數(shù)據(jù) ———— 希望開小點(diǎn)
?? ?struct Stack st2; ?
?? ?StackInitCpp2(&st1);//缺省
?? ?return 0;
}二、缺省參數(shù)分類
? 全缺省參數(shù) ?
void TestFunc(int a = 10, int b = 20, int c = 30)
{
?? ?cout << "a = " << a << endl;
?? ?cout << "b = " << b << endl;
?? ?cout << "c = " << c << endl;
?? ?cout << endl;
}
int main()
{
?? ?//非常靈活,
?? ?TestFunc();
?? ?TestFunc(1);
?? ?TestFunc(1, 2);
?? ?TestFunc(1, 2, 3);?? ?
?? ?//TestFunc(1, , 3);//err,注意它沒辦法實(shí)現(xiàn)b不傳,只傳a和b,也就是說編譯器只能按照順序傳
?? ?return 0;
}? 注意:
1?? 全缺省參數(shù)只支持順序傳參
? 半缺省參數(shù) ?
//void TestFunc(int a, int b = 10, /*int f, - err*/ int c = 20);//err,
void TestFunc(int a, int b = 10, /*int f, int x = y, -> err*/ int c = 20)
{
?? ?cout << "a = " << a << endl;
?? ?cout << "b = " << b << endl;
?? ?cout << "c = " << c << endl;
?? ?cout << endl;
}
int main()
{
?? ?//TestFunc();//err,至少得傳一個(gè),這是根據(jù)形參有幾個(gè)非半缺省參數(shù)確定的
?? ?TestFunc(1);
?? ?TestFunc(1, 2);
?? ?TestFunc(1, 2, 3);?? ?
?? ?return 0;
}//a.h
void TestFunc(int a = 10);
//a.cpp
void TestFunc(int a = 20)
{}
? 注意:
1?? 半缺省參數(shù)必須從右往左依次來給出,且不能間隔著給
2?? 缺省參數(shù)不能在函數(shù)聲明和定義中同時(shí)出現(xiàn)
3?? 缺省值必須是常量或者全局變量
4?? C 語言不支持缺省
缺省參數(shù)的誤區(qū)
使用缺省參數(shù)時(shí)應(yīng)該注意避開下列幾種誤區(qū)。
1.濫用缺省參數(shù),損害代碼的結(jié)構(gòu)和可讀性。
void f(bool b=false)
{
if (b)
{
file://code of open file
}
else
{
file://code of close file
}
}
打開文件和關(guān)閉文件在實(shí)現(xiàn)代碼上沒有什么共同點(diǎn),把兩個(gè)屬于同一類別的函數(shù)誤認(rèn)為是實(shí)現(xiàn)機(jī)制相同,憑空捏造一個(gè)參數(shù)硬把它們湊在一塊,沒有什么好處!相反,誰能記得住f(true)代表打開,f()代表關(guān)閉呢?況且,f(false)、f()都可以關(guān)閉文件,如果調(diào)用者混合使用它們就會(huì)增加維護(hù)上的困難。這種情況下,寫成兩個(gè)獨(dú)立的函數(shù),非常清晰。
void Open()
{
file://code of open file
}
void Close()
{
file://code of close file
}
推而廣之,如下的做法也值得商榷。
class CString
{
private:
char * pcData;
public:
CString(char * pc=NULL);
};
CString::CString(char * pc)
{
if (pc==NULL)
{
pcData=new char[1];
//...
}
else
{
pcData=new char[strlen(pc)+1];
//...
}
}
這一個(gè)更具備迷惑性,“都是構(gòu)造器嘛,當(dāng)然寫在一塊嘍。”有人說。非也!應(yīng)當(dāng)看到,無參構(gòu)造器與帶char *參數(shù)的構(gòu)造器使用的代碼完全分離,并且缺省參數(shù)值NULL在設(shè)置數(shù)據(jù)成員時(shí)沒有任何作用。CString()構(gòu)造器應(yīng)改寫如下:
class CString
{
private:
char * pcData;
public:
CString();
CString(char * pc);
};
CString::CString()
{
pcData=new char[1];
//...
}
CString::CString(char * pc)
{
pcData=new char[strlen(pc)+1];
//...
}
總結(jié):
(1)凡是出現(xiàn)利用缺省參數(shù)值作if判斷,并且判斷后實(shí)現(xiàn)代碼完全不同的,都應(yīng)該分拆成兩個(gè)獨(dú)立的函數(shù)。
(2)只有缺省參數(shù)值在函數(shù)體中被無歧視的對待,也就是函數(shù)對于任何參數(shù)的實(shí)現(xiàn)機(jī)制都相同時(shí),才可能是合理的。
2.多個(gè)缺省參數(shù),可能引入邏輯含混的調(diào)用方式
設(shè)計(jì)一個(gè)類,不僅僅是提供給客戶代碼正確的功能,更重要的是,對不正確的使用方式作力所能及的限制。
class CPoint
{
public:
int x;
int y;
CPoint(int x=0,int y=0)
{
this->x=x;
this->y=y;
}
};
乍一看,沒什么問題。構(gòu)造CPoint對象時(shí)如果不指定x、y的初值,則設(shè)為原點(diǎn)坐標(biāo)。讓我們測試一下:
CPoint pnt1;
CPoint pnt2(100,100);
CPoint pnt3(100); file://[1]
結(jié)果發(fā)現(xiàn)pnt3的值為(100,0),跑到x軸上去了。對于想綁定兩個(gè)參數(shù),讓它們同時(shí)缺省,或者同時(shí)不缺省,我們無能為力。但是如果去掉缺省參數(shù),情況就會(huì)好轉(zhuǎn)。
class CPoint
{
public:
int x;
int y;
CPoint()
{
x=0;
y=0;
}
CPoint(int x,int y)
{
this->x=x;
this->y=y;
}
};
這樣,語句[1]就會(huì)引發(fā)編譯錯(cuò)誤,提醒使用者。
抬杠的會(huì)說:“CPoint pnt3(100);初始化到x軸,本來就是我想要的。”真的嗎?那么,請你在你的類文檔中明確指出這種獨(dú)特的調(diào)用方法,并且告訴使用者,將點(diǎn)初始化到y(tǒng)軸是CPoint pnt4(0,100);這種不對稱的形式。
至于我嘛,self document好了。
3.重載時(shí)可能出現(xiàn)二義性
這個(gè)簡單,隨便舉個(gè)例子:
void f(int a,int b=0)
{
}
void f(int a)
{
}
雖然潛在的模棱兩可的狀態(tài)不是一種錯(cuò)誤,然而一旦使出現(xiàn)f(100);這樣的代碼,潛伏期可就結(jié)束了。
4.函數(shù)調(diào)用中的精神分裂癥
Effective C++ 2nd中的條款,為了本篇的完整性加在這里。這種罕見的癥狀出現(xiàn)的條件是:派生類改寫了基類虛函數(shù)的缺省參數(shù)值。
class CBase
{
public:
virtual void f(int i=0)
{
cout<<"in CBase "<<i<<endl;
}
};
class CDerive : public CBase
{
public:
virtual void f(int i=100)
{
cout<<"in CDerive "<<i<<endl;
}
};
CDerive d;
CBase * pb=&d;
pb->f(); file://[2]
運(yùn)行后輸出:
in CDerive 0
記住,缺省參數(shù)是靜態(tài)綁定,而虛函數(shù)是動(dòng)態(tài)綁定,所以[2]運(yùn)行的是CDerive::f()的函數(shù)體,而使用的缺省值是CBase的0。
到此這篇關(guān)于C++缺省參數(shù)的具體使用的文章就介紹到這了,更多相關(guān)C++缺省參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ abs函數(shù)實(shí)際應(yīng)用詳解
本文我們來講C++的abs函數(shù)以及實(shí)戰(zhàn)運(yùn)用,C++中的abs函數(shù)。在C++中使用abs函數(shù)要注意存在兩種版本,一種是在stdlmb.h中定義的版本,另一個(gè)是在cmath頭文件中定義的。夷實(shí)上在stdlib.h文件是C的函數(shù),而cmath中的是C++版本2022-08-08
C/C++開發(fā)中extern的一些使用注意事項(xiàng)
這篇文章主要為大家介紹了C/C++開發(fā)中extern一些使用注意事項(xiàng)的事例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
C++中成員函數(shù)和友元函數(shù)的使用及區(qū)別詳解
大家好,本篇文章主要講的是C++中成員函數(shù)和友元函數(shù)的使用及區(qū)別詳解,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下2022-01-01
VScode配置cuda開發(fā)環(huán)境的實(shí)現(xiàn)步驟
本文主要介紹了VScode配置cuda開發(fā)環(huán)境的實(shí)現(xiàn)步驟,文中通過圖文介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-07-07

