作者:陳太漢
是的。我們討厭產生臨時變數,因為它要佔用我們的記憶體,消耗我們的CPU時間,讓我們的程式效能降低。但有時候它是必須的。
那討厭的臨時變數什麼時候產生呢?
產生臨時變數的三種情況:一:以By Value的方式傳值;二:參數為const的類型。三:類型轉換
一:以By Value的方式傳值。
我們都知道,參考型別和指標類型傳遞的都是地址,可以直接對地址中存放的資料進行操作,
而以傳值的方式傳遞參數,就會在heap中重新分配一個臨時地區,
將實參中的資料拷貝到臨時地區中,而你對這分資料進行的任何的操作都不會影響實參的內容,因為實參跟形參只是內容相同,
分別在兩塊不同的記憶體中。而引用和指標操作的是同一塊記憶體,所以形參修改後,實參也修改了。
二:參數為const的類型。(此觀點錯誤,請看我的這篇文章:Effective C++ 類與函數的設計和申明)
因為常量是不能修改,在只需要實參中的資料,而不需對實參進行修改時,或是禁止對實參進行修改時,把形參定義為const類型,
系統會產生一個臨時變數,就能起到保護資料的作用,如在函數strlen中,修改參數的值行嗎?本來只是想得到實參的長度,結果在函數中被修改了,那得到得實參長度還是真實的嗎。
如果你程式中的資料到處都可以被修改,那是多麼的可怕(所以我們討厭全域變數),所以const還是有它存在的價值。
三:類型轉換的時候會產生臨時變數。
真是糟糕啊,在用類型轉換帶來便利的同時,產生臨時變數就是我們承擔的損失。
如將一個short類型轉換成int類型,他們佔用的記憶體不一樣,如果不產生臨時變數,那不就short類型和int類型佔用的位元組數不就一樣了嗎,sizeof不就坑爹了嗎
C++語言禁止為非常量引用產生臨時對象。同時證明參考型別傳參不會產生臨時變數,如char[]轉換成string會報錯,他們都是參考型別
以下是我自己寫的基本函數,練手啊,下面這些函數參數前面很多都加了const,那不是到處都在產生臨時變數嗎?我在想,我就不要const,我就不準對這些參數進行任何修改,行嗎?
誰能給我答案?我只想在不需要產生臨時變數的時候不讓它產生......跟著標準走吧。
基本函數
//字串賦值之前先delete左邊那個字串,再分配一塊新的記憶體的記憶體存放右邊的字串,讓左邊字串的指標指向這塊新記憶體
//如果左右兩邊的字串相等,即指向同一塊記憶體,delete左邊的字串相當於delete了這兩個字串。
char* strcpy(char* str,constchar* source)
{
if(str==source)
{
return str;
}
if(str==NULL || source==NULL)
{
return NULL;
}
char* ptStr=str; //為了實現串聯操作
while((*str++=*source++)!=NULL);
return ptStr;
}
//參數為const類型,系統會產生一個臨時變數用於存放實參的內容,
//但如果形參不定義成const類型,如果參數的內容被修改了,那麼這個函數可能就不能得到正確的值.
int strlen(constchar* str)
{
if(NULL==str)return-1;
int len=-1;
while(*(str+(++len))!='\0');
return len;
}
//指定字元第一次出現的位置
constchar* strchr(constchar* str,char c)
{
if(NULL==str)
{
return NULL;
}
while(str &&*str!=c)
{
str++;
}
if(str=='\0')
{
return NULL;
}
return str;
}
//子字串第一次出現的位置
int strstr(constchar* str,constchar* substr)
{
if(NULL==str || NULL==substr)
{
return-1;
}
if(strlen(str)<strlen(substr))
{
return-1;
}
constchar* tmp;
int i=0;
while(*str!='\0')
{
tmp=substr;
if(*str==*tmp)
{
int j=0;
while(*str!='\0'&&*tmp!='\0'&&*(str+j++)==*tmp++);
if(*tmp=='\0')
{
return i;
}
}
str++;i++;
}
return-1;
}
//追加字串
char* strcat(char* str,constchar* cat)
{
if(NULL==cat || NULL==str)
{
return str;
}
char* tmp=str;
while(*str)//將指標移到最後
{
str++;
}
while((*str++=*cat++)!='\0');//追加字元
str=tmp;
return str;
}
//比較字串
int strcmp(constchar* str1,constchar* str2)
{
if(NULL==str1 && NULL==str2)
{
return0;
}
if(NULL==str1)
{
return-1;
}
if(NULL==str2)
{
return1;
}
while(*str1 &&*str2 &&*str1==*str2)
{
str1++;
str2++;
}
return*str1-*str2;
}