第4章 數組、指針與引用
4.1 數組
1一維數組
(1)一維數組的定義
一維數組是由具有一個下標的數組元素組成的數組,其定義形式如下:
<數據類型><數組名>[<數組長度>];
在此,<數據類型>是類型說明符,<數組名>是數組的名字,<數組長度>是任一值為正整數的int型常量表達式。<數組長度>用來指定數組中元素的個數,即數組的大小,數組元素的下標是從0到<數組長度>-1。
(2)一維數組初始化
與所有的基本數據類型相同,數組也可以在時初始化。
當數組聲明沒有給出數組的大小,但是有初始化列表時,數組的大小就由列表中元素的個數來確定。
在定義數組時,可以不必給出所有數組元素的初始值,即在定義時部分地初始化數組。
(3)訪問數組元素
訪問數組元素的語法形式是:
<數組名>[<表達式>]
在此,<表達式>是非負的int型表達式,稱為下標。下標用于指定所要訪問的數組中元素的位置。在C+ +中,[]是一個運算符,稱為下標運算符。數組下標從0開始,長度為n的數組,其下標的范圍是0到n-1。在數組定義以后,給數組賦值時,必須一個元素一個元素的逐個訪問。
2二維數組
以行和列(即二維)形式排列的固定數目元素的集合,并且組成數組的每個元素的類型都相同,即帶有兩個下標的數組。
(1)二維數組的定義
定義二維數組的語法是:
<數據類型><數組名>[<表達式1>][<表達式2>];
在此<表達式1>和<表達式2>是值為正整數的常量表達式。表達式<表達式1>和<表達式2>,分別用來指定數組中行和列的數目。
(2)二維數組初始化
與一維數組相同,二維數組也可以在定義時初始化。
在定義時初始化二維數組要注意:
①所有在一行中的元素可以用花括號括起來,并且用逗號分隔;
②對于數字類型數組,如果給定的數值不夠,則沒有指定數值的元素將初始化為0。此時,至少應有一個值來初始化二維數組。
③當數組聲明時沒有給出第一維的大小,但是有初始化列表時,第一維的大小就由列表中子列表的個數來確定。
(3)訪問二維數組元素
若要訪問二維數組的元素,必須要給出兩個下標:一個行下標和一個列下標。訪問二維數組元素的語法是:
<數組名>[<表達式1>][表達式2];
這里<表達式1>和<表達式2>是值為非負整數的表達式。<表達式1>指定行下標,<表達式2>指定列下標。
(3)多維數組
在C+ +中,對數組維數沒有限制。下面是對數組的通用定義。
以n維列表形式排列的固定數目元素的集合,稱為n維數組。定義n維數組的語法是:
<數據類型><數組名>[<表達式1>][<表達式2>]…[<表達式n>];
在此<表達式1>,<表達式2>,…<表達式n>都是值為正整數的常量表達式。
訪問n維數組元素的語法是:
<數組名>[<表達式1>][<表達式2>]…[<表達式n>];
在此<表達式1>,<表達式2>,…<表達式n>都是值為非負整數的表達式。<表達式i>(i=1,2,…,n)用來指定第i維元素的位置。
在將多維數組作為函數的形參時,可以不指定該數組中第一維的大小,但是必須指定該數組中其他維的大小。多維數組只可以作為引用參數傳遞給函數,并且函數不能返回一個數組類型的返回值。C+ +對數組下標不做檢查。
4字符數組
(1)string類型與字符數組
string類型在ANSI/ISO標準C+ +頒布之前,標準C+ +庫并不提供string數據類型。string是一種用戶自定義的數據類型,它由C+ +標準庫來支持,而不是C+ +語言本身的一部分。在使用strign數據類型之前,需要在程序中包含頭文件string并聲明其所在的名字空間std。所有元素都是char類型的數組稱為字符數組。
字符數組有如下特點:
①數組元素跟一般變量一樣可以賦值、比較、計算等。
②數組下標也是從0~N-1(N為數組長度)。
③字符數組長度可以顯式給出,也可以隱式得到。
④由雙引號括起來的字符串常量具有靜態字符串數組類型。
⑤用字符串對數組初始化時,編譯程序以′\0′作為結束這個數組的標志。因此,數組長度至少要比字符串長度多1。
(2)常用字符串函數
C+ +提供了一系列字符串操作的函數,這些函數都包含在頭文件cstring中。其中經常會用到的字符串函數包括:strcpy(字符串拷貝,將一個字符串拷貝到另一個字符串變量中)、strcat(字符串連接,在字符串末端添加字符串)、strcmp(字符串比較,用來比較字符串)、strlen(字符串長度,用來求出字符串長度)、和strstr(字符串查找,在一個字符串中查找子串)。
①strcpy函數
調用strcpy()函數的一般形式為:
strcpy(to,from);
該函數將from字符串中的內容復制到to字符串中。請記住,構成to的字符串必須足夠大,以便保存包含在from中的字符串。否則,to字符串將會溢出,這很可能會導致系統崩潰。
②strcat函數
調用strcat()函數的一般形式為:
strcat(s1,s2);
該函數將字符串s2添加到字符串s1的末端;但并不修改字符串s2。必須確保字符串s1足夠大,以便保存它自己的內容和字符串s2中的內容。
③strcmp函數
調用strcmp()函數的一般形式為:
strcmp(s1,s2);
該函數比較兩個字符串,如果兩個字符串相等,返回0。如果字符串s1在字典順序上比字符串s2大,則返回一個正數;如果比字符串s2小,則返回一個負數。
注意:函數strcmp比較兩個字符串時所采用的字典順序與真正意義上的字典順序還是有些差別的。實際上,它依次比較兩個字符串相應字符的編碼值(如ASCII碼值)來決定兩個字符串的大小。如,上面比較strcmp(″A″,″a″)的結果變為-1,因為字符′A′的編碼值為65,而字符′a′的編碼值為97,這與傳統意義上的字典順序不同。
④strlen函數
調用strlen()函數的一般形式為:
strlen(s);
該函數返回字符串s的長度,即字符串中字符的個數(不包括字符串結尾的′\ 0′)。
⑤strstr函數
調用strstr()函數的一般形式為:
strstr(s1,s2);
該函數在字符串s1中從左邊開始查找字符串s2,若查找成功則返回s2在s1中首次出現的位置,否則返回NULL,如果s2為″ ″,則返回s1。
4.2 指針
指針(即指針變量)是C+ +語言最大的功能之一。一個指針是一個特定類型數據的存儲地址,比如一個變量的地址。
與其他類型變量一樣,指針變量也必須要聲明。指針變量聲明的一般形式為:
<數據類型>*<變量名>;
其中,<數據類型>是指針所指對象的類型,在C+ +中可以指向任何C+ +類型。<變量名>是指針變量名。
指針使用兩種特殊的運算符——*和&。
一元(單目)運算符&用于返回其操作對象的內存地址,其操作對象通常為一個變量名。例如:
ptr=&total;
把變量total的內存地址存到指針變量ptr中。該地址是total變量在計算機內存中的存儲地址。
第二個與指針相關的運算符是*,它與&運算符作用相反。作為一元運算符的*用于返回其操作數所指對象的值,因此,該運算符要求其操作對象為一個指針。
1指針和地址
(1)指針說明
從指針的定義可知,指針是用所指對象類型來表征的。在使用任何指針變量之前必須先給它賦一個所指合法具體對象的地址值。
如何使一個指針指向一個具體對象:
①使用new運算符(或malloc和alloc等函數)給指針分配一個具體空間。
②將另一個同類型的指針賦給它以獲得值。
③通過&運算符指向某個對象。
(2)指針運算
盡管指針中存放的是變量的地址,但在C+ +中指針只能進行如下運算。
①指針和整型量可以進行加減
②若p1,p2為指針,當p1和p2指向同一類型時,可以進行賦值。
③兩個指向同一類型的指針,可進行= =,>,<等關系運算,其實就是地址的比較。
④兩個指向同一數組成員的指針可進行相減,結果為兩個指針之間相差元素的個數。
注意:兩指針不能相加。
2指針和數組
在C+ +中,指針和數組的關系極為密切。實際上,數組的參數傳遞、數組元素的存取,都可通過指針操作來完成。指針和數組常常可以互換。
在C+ +中,數組的名字就是指向該數組第一個元素(下標為0)的指針,即該數組第一個元素的地址,也即數組的首地址。
一般情況下,一個數組元素的下標訪問a[i]等價于相應的指針訪問*(a+i)。但特別注意:數組名和指針(變量)是有區別的,前者是常量,即數組名是一個常量指針,而后者是指針變量。因此,盡管我們可寫pa=a;但不能寫:a=pa;或pa=&a;因為我們不能改變常量的值,也不能取常量的地址。
數組名可作為參數進行傳遞。當將數組名傳給函數時,實際上所傳遞的是數組的開始地址。(即數組第一個元素的地址)
為什么要使用指針?簡單地說指針運算比數組運算的速度快。
此外,使用指針的另外一個原因是在大量數據傳遞時,使用傳遞指針要遠比傳遞數據本身效率高的多,如在函數參數傳遞及函數返回值時。當然,使用指針會給程序帶來安全隱患(如指針懸掛問題),同時還使得程序的可讀性降低(顯然,用數組實現的程序要比用指針實現的程序的可讀性要好)。
對于字符串常量,可以把它看成是一個無名字符數組,C+ +編譯程序會自動為它分配一個空間來存放這個常量,字符串常量的值本身就是指向這個無名字符數組的第一個字符的指針,其類型是字符指針。
3指針數組和函數指針
(1)指針數組
指針數組就是由指針組成的數組,即數組中的每一個元素都是指向同一類型對象的指針。
指針數組可以是全局的、靜態的和局部的。字符指針數組和二維字符數組在許多方面是一樣的,如初始化形式、成員訪問方式,因此我們有必要了解它們的區別和各自的使用場合。
盡管二維字符數組與字符指針數組在存儲形式上不同,但它們在初始化形式以及訪問元素方式上卻是相同的。
采用指針數組的理由是:它可以節省存貯空間,因而通常用來存放不同長度的字符串。例如,如果要保存從標準輸入或文件中讀入的行,字符指針數組是一個好的選擇。因為讀入的行可能長短差異很大。
(2)命令行參數
在C+ +中可以實現帶有命令行參數的程序,它是通過main帶有參數來實現的。
在C+ +中,主函數main還可以帶有參數,形式如下:
int main(int argc,char * argv[])
或int main(int argc,char * * argv[])
其中:
argc為包含命令本身在內的參數個數。
argc為指針數組,數組元素為指向各參數(包含命令本身在內)的指針。
(3)函數指針
在C+ +中,允許指針指向一個函數,即指向函數的指針。函數指針的說明形式為:
類型(*標識符)();
例如:int (* fp)();定義了一個指向返回值為整型值的函數的指針fp。
注意:int(*fp)();與int*fp();的不同在于,前者是函數指針,后者為返回指向int指針的函數。
與其他類型的指針變量一樣,在使用函數指針前必須使它指向一個具體的函數。若要函數指針指向一個具體函數,可通過賦值語句或參數傳遞。
函數指針=函數名;
該賦值語句將使一個函數指針指向一個具體函數(在C+ +中,函數名是作為指向函數的指針值來處理)。函數指針的最大用途是它可以使得一個函數作為其他函數的參數進行傳遞,擴展了函數的功能。
4.3 引用
1引用的概念
引用是個別名,建立時須用另一個數據對象(如一個變量)的名字進行初始化,以指定該引用所代表的數據對象。此后,對引用的任何實操作實際上就是對所代表的數據對象的操作。一個引用變量要占用相當于一個指針所需要的空間,但系統不會為它所代表的數據對象再次分配空間。
在類型名后跟引用運算符“&”,以及引用名來創建一個引用。引用名就是一個變量名。
注意:引用運算符與地址操作符使用相同的符號(即運算符重載),但它們含義不一樣。引用運算符只在聲明變量的時候使用,它放在類型名后面。
使用引用時應遵循一定的規則:
(1)引用被創建時,它必須立即被初始化(指針則可以在任何時候被初始化)。
(2)一旦一個引用被初始化為一個對象的引用,它就不能再被改變為對另一個對象的引用。(指針則可以在任何時候改變為指向另一個對象。)
(3)不可能有NULL引用。必須確保引用是具體合法的對象的引用(即引用應和一塊合法的存儲空間關聯)。
2用引用傳遞函數參數
引用的一個重要用途就是作為函數的參數。在C+ +中,函數參數傳遞采用的是傳值,如果要有占用空間大的對象(例如一個大的結構對象或類對象)需要作為函數參數傳遞的時候,在C語言中的做法往往是使用指針,因為這樣可以避免將整個實參對象數據全部拷貝給形式參數,可以提高程序的執行效率。在C+ +中,既可以使用指針,但由于C+ +引入了引用概念,亦可以用引用來做同樣的事情。引用作為參數的最大好處是:引用參數既可以像指針那樣工作,其使用方式又和一般變量相同。也就是說,引用比指針具有更好的可讀性。
4.4 動態存儲分配
動態存儲分配功能在C+ +中是通過new和delete運算符來實現的。
1使用new獲得動態內存空間
運算符new用于申請動態存儲空間,它的操作數為某種數據類型且可以帶有初值表達式或元素個數。new返回一個指向其操作類型變量的指針。使用new對某種類型變量進行動態分配的語法格式為:
<指針>=new<類型>;
其中,<類型>表示要分配的變量類型(如char、int、double);<指針>表示指向<類型>類型變量的指針(如char*、int*、double*等)。
2使用delete釋放動態內存空間
當動態分配的內存空間在程序中使用完畢之后,必須顯式地將它們釋放。這樣做的目的是把閑置不用的堆內存歸還給系統,使其可以被系統重新分配。在C+ +程序中由new分配的動態內存空間必須通過delete運算符釋放。使用delete對動態分配的單個變量進行釋放的語法格式為:
delete<指針>;
其中,<指針>表示指向單個變量的指針。
使用delete對動態分配的數組進行釋放的語法格式為:
delete[]<指針>;
其中,<指針>表示指向數組首元素的指針。delete之后的方括號指明將要釋放的內存空間中存儲著數組元素。程序中需要分配動態內存空間,則new和delete總是成對出現的。
北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內蒙古 |