版權說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權,請進行舉報或認領
文檔簡介
1、第12章指針的深入研究——指針進階,指針與數(shù)組 指針與字符串 綜合應用——報數(shù)游戲 跟我上機,第12章指針的深入研究——指針進階,指針與數(shù)組 指針與字符串 綜合應用——報數(shù)游戲 跟我上機,第12章指針的深入研究——指針進階,指針與數(shù)組 指針與字符串 綜合應用——報數(shù)游戲 跟我上機,第12章指針的深入研究——指針進階,指針與數(shù)組 指針與字符串 綜合應用——報數(shù)游戲 跟我上機,第12章
2、指針的深入研究——指針進階,指針與數(shù)組 指針與字符串 綜合應用——報數(shù)游戲 跟我上機,12.1 指針與數(shù)組,12.1.1 指針與數(shù)組12.1.2 數(shù)組名作為指針12.1.3 指針與多維數(shù)組12.1.4 指針與數(shù)組參數(shù),12.1.1 指針與數(shù)組,指針用于存儲地址,他和數(shù)組名緊密的聯(lián)系在一起。下圖就顯示了一個數(shù)組名稱為grade,包含5個整型數(shù)據(jù)的一維數(shù)組,這里每個整型元素占用4個字節(jié)。,,12.1.1
3、指針與數(shù)組,訪問數(shù)組元素,我們可以使用下標來實現(xiàn),比如我們要訪問數(shù)組元素grade[3],在之前的章節(jié)里我們詳細的講解過,但是那時使用的方法其實隱藏了數(shù)組每一個元素的地址。根據(jù)之前講過的地址的概念,加之我們已知的數(shù)組的特點,這個特點就是數(shù)組在內(nèi)存總占用一塊連續(xù)的存儲區(qū)域,這樣的話,如果我們知道數(shù)組的首地址,針對grade這個數(shù)組,就是已知grade[0]的地址,我們就可以通過增減偏移量,得到garde[3]的地址,從而訪問到garde[
4、3]的元素值,12.1.1 指針與數(shù)組,&grade[3]=&grade[0] + 3 * 4上面表達式的含義是gradep[3]的地址等于grade[0]地址加上12。寫成以下的通式:數(shù)組中下標為N元素的地址 = 數(shù)組的首地址 + N * sizeof(DataType)例如:int grade[5]; /*定義array為包含5個整型數(shù)據(jù)的數(shù)組*/int *p; /*定義p為指向整型變量的指針變量*/
5、p=&grade[0];,,12.1.1 指針與數(shù)組,【范例12-1】 使用指針訪問數(shù)組元素,分別使用下標和指針訪問數(shù)組元素01 #include 02 int main(void)03 {04 int i;05 int grade[]={2,5,9,4,6};06 int *p;07 p=&grade[0]; /*指針指向*/08 print
6、f("使用下標訪問數(shù)組元素:\n");09 for(i=0;i<5;i++) /*下標訪問*/10 printf("數(shù)組grade的第%d個元素是%d\n",i,grade[i]);11 printf("使用數(shù)組指針訪問數(shù)組元素:\n");12 for(i=0;i<5;i++) /*指針訪問*/13
7、printf("數(shù)組grade的第%d個元素是%d\n",i,*(p+i));14 return 0;15 },12.1.1 指針與數(shù)組,需要注意的是*(p+i)表達式中的括號是必須有的,不能省略,如果我們遺漏了小括號,將變成這樣的表達式*p+i,它的含義之前在指針變量處已經(jīng)進行了講解,是在p所指向的存儲單元的數(shù)值基礎上在i,因為我們始終沒有改變指針p的指向。,12.1.2 數(shù)組名作為指針,每個創(chuàng)
8、建的數(shù)組,數(shù)組名就成為編譯器為這個數(shù)組所創(chuàng)建的指針常量名稱,存儲的是數(shù)組第一個元素的起始地址,也是我們所說的數(shù)組首地址。這樣,我們就有多了一種獲取數(shù)組元素地址的手段。如【范例12-1】中代碼“p=&grade[0];”就可以改寫為“p=grade;”。,12.1.2 數(shù)組名作為指針,【范例12-2】 使用數(shù)組名作為指針訪問數(shù)組元素。01 #include 02 int main(void)03 {04
9、 int i;05 int grade[]={2,5,9,4,6};06 int *p;07 p=grade; /*指針賦值*/08 printf("使用數(shù)組名訪問數(shù)組元素:\n");09 for(i=0;i<5;i++) /*使用數(shù)組名*/10 printf("數(shù)組grade的第%d個元素是%d\n",i,*(gr
10、ade+i));11 printf("使用數(shù)組指針訪問數(shù)組元素:\n");12 for(i=0;i<5;i++) /*使用指針變量*/13 printf("數(shù)組grade的第%d個元素是%d\n",i,*(p+i));14 return 0;15 },12.1.2 數(shù)組名作為指針,面的代碼大家一起來分析一下,看看錯了嗎?grade =
11、 grade+3;結(jié)果是有誤。原因是,grade是數(shù)組名,它是編譯器為這個數(shù)組創(chuàng)建的指針常量,常量當然是不能夠再賦值的。再看下面的代碼,是否有問題?p = grade;p = p+3;結(jié)果是正確的。原因是,p是指針變量,它初始值是grade數(shù)組的首地址,之后根據(jù)需要修改了p的值,也就是改變了指針變量p的指向。,12.1.2 數(shù)組名作為指針,【范例12-3】 已知一個含有10整型元素的一維數(shù)組array,使用指針,求出該數(shù)組的
12、最大值,最小值和平均值,并輸出。,12.1.2 數(shù)組名作為指針,【范例12-4】有n個整數(shù),存儲在數(shù)組array中,使其前面各數(shù)順序向后移m個位置,最后m個數(shù)變成最前面的m個數(shù),并輸出。,12.1.3 指針與多維數(shù)組,我們知道數(shù)組在內(nèi)存中用連續(xù)的存儲區(qū)域,這一點已經(jīng)反復提到,我們以二維數(shù)組為例,數(shù)據(jù)是怎么存儲的呢?假設存在二維數(shù)組array[2][3],那么這6個數(shù)據(jù)在內(nèi)存中按照行走Z形排列,在內(nèi)存儲中存儲形式如下圖所示。,,12.
13、1.3 指針與多維數(shù)組,多維數(shù)組的數(shù)組名也被編譯器創(chuàng)建為常量,用來存儲該數(shù)組的首地址。數(shù)組名array存儲的是array[0][0]元素的地址,也就是存儲的值是&array[0][0]。我們也可以按照首地址加偏移量的形式,從而使指針指向該數(shù)組中任何一個元素,實現(xiàn)多維數(shù)組和指針的關聯(lián)。,12.1.3 指針與多維數(shù)組,下面我們逐步細分,讓大家掌握。第一步,我們寫出最容易理解的形式:array[1][2]/*指的是一個元素*
14、/&array[1][2]/*指的是array[1][2]元素的地址*/然后,我們深入一步:array/*指的是數(shù)組下標為0的那一行的首地址*/array+1/*指的是數(shù)組下標1的那一行的首地址*/&array[1]/*指的也是數(shù)組下標1的那一行的首地址*/,12.1.3 指針與多維數(shù)組,接下來,我們進一步深入:array[1]/*等價array[1]+0,于指的也是array[1][0]元素的地址
15、*/array[1]+2/*指的也是array[1][2]元素的地址*/*(array+1)+2/*指的也是array[1][2]元素的地址*/最后:*(*(array+1)+2)/*指的是元素array[1][2]值*/*( array[1]+2)/*指的是元素array[1][2]值*/,12.1.3 指針與多維數(shù)組,【范例12-5】 使用指針訪問二維數(shù)組,12.1.4 指針與數(shù)組參數(shù),1. 第一種情況:實參
16、是數(shù)組,形參也是數(shù)組主調(diào)函數(shù):int arr[10];function(arr);被調(diào)函數(shù):function(int arr[]) {...},12.1.4 指針與數(shù)組參數(shù),2. 第二種情況:實參是數(shù)組,形參是數(shù)組指針主調(diào)函數(shù):int arr[10];function(arr);被調(diào)函數(shù):function(int *p) {...},12.1.4 指針與數(shù)組參數(shù),3. 第三種情況:實參是數(shù)組指針,形參是數(shù)組指針。
17、主調(diào)函數(shù):int arr[10];int *p=arr;function(p);被調(diào)函數(shù):function(int *q) {...},12.1.4 指針與數(shù)組參數(shù),4. 第四種情況:實參是數(shù)組指針,形參是數(shù)組。主調(diào)函數(shù):int arr[10];int *p=arr;function(p);被調(diào)函數(shù):function(int arr[]) {...},12.1.4 指針與數(shù)組參數(shù),【范例12-6】 統(tǒng)計在a
18、rray字符串中從'a'到'z'的26個小寫字母各自現(xiàn)的次數(shù)。例如:輸入字符串“abcdefgabcdeabc”,輸出的結(jié)果應該是“3 3 3 2 2 2 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0”,12.2 指針與字符串,12.2.1 使用指針創(chuàng)建字符串12.2.2 使用指針處理字符串12.2.3 指針的指針12.2.4 指針數(shù)組和數(shù)組指針,1
19、2.2.1 使用指針創(chuàng)建字符串,字符串的定義自動就包含指針,例如,我們定義message1[100];為100個字符聲明存儲空間,并自動的創(chuàng)建一個包含指針常量message1,是存儲的是message1[0]的地址。跟一般的常量一樣,指針常量指向是明確的,不能被修改。對于字符串,我們可以不按照聲明一般數(shù)組的方式,定義數(shù)組維數(shù)和每一維的個數(shù),還可以使用新的方法,也就是我們要講解的用指針創(chuàng)建字符串。例如下面的代碼:char *mes
20、sage2="how are you?";這個message1和message2是不同的,message1是按照數(shù)組定義方式定義的,如:char message1[100]= "how are you?";,12.2.1 使用指針創(chuàng)建字符串,這種形式要求,message1有固定的存儲該數(shù)組的空間,而且,因為message1本身是數(shù)組名稱,一旦被初始化后,再執(zhí)行下面的語句就是錯誤的,如
21、message1= "fine,and you?";message2本身就是一個指針變量,它通過顯式的方式明確了一個指針變量,對于message2執(zhí)行了初始化后,再執(zhí)行下面的代碼是正確的,message2= "fine,and you?";從分配空間的角度來分析,二者也是不同的。message1指定了一個存儲100個字符位置的空間,而對于message2就不同了,它只能存儲一個地
22、址,它只能保存指定字符串的第一個字符的地址。,12.2.1 使用指針創(chuàng)建字符串,【范例12-7】 八進制轉(zhuǎn)十進制。01 #include 02 int main(void)03 {04 char *p,s[6];05 int n;06 n=0;07 p=s;/*字符指針p指向字符數(shù)組s*/08 printf("輸入你要轉(zhuǎn)換的八進制數(shù):\n");
23、09 gets(p);/*輸入字符串*/10 while(*(p)!='\0')/*檢查指針是否都字符數(shù)組結(jié)尾*/11 {12 n=n*8+*p-'0';/*八進制轉(zhuǎn)十進制計算公式*/13 p++;/*指針后移*/14 }15 printf("轉(zhuǎn)換的十進制是:\n%d\n",n);16 re
24、turn 0;17 },12.2.2 使用指針處理字符串,【范例12-8】 字符串復制01 #include 02 int main(void)03 { 04 char str1[10],str2[10];05 char *p1,*p2;06 p1=str1;07 p2=str2;08 printf("請輸入原字符串:\n");09
25、 gets(p2);10 for (; *p2!='\0';p1++,p2++) /*循環(huán)復制str2中的字符到str1*/11 *p1=*p2;12 *p1='\0'; /*str1結(jié)尾補\0*/13 printf("原字符串是 :%s\n復制后字符串是:%s\n",str2,str1);14 return 0;15
26、 },12.2.2 使用指針處理字符串,這道題目聲明了兩個字符串的指針,通過指針移動,賦值字符串str2中的字符到str1,并且在str1結(jié)尾添加了字符串結(jié)束標志。需要說明以下兩點:⑴ 如果題目中沒有使用指針變量,而是直接在for循序中使用了str1++這樣的表達式,程序就會出錯,因為str1是字符串的名字,是常量;⑵ 如果沒有寫*p1='\0';這行代碼,輸出的目標字符串長度是9位,而且很可能后面的字符是
27、亂碼,因為str1沒有結(jié)束標志,直至遇見了聲明該字符串時設置好的結(jié)束標志“\0”。,12.2.2 使用指針處理字符串,【范例12-9】 字符串連接。,12.2.2 使用指針處理字符串,【范例12-10】 已知一個字符串,使用返回指針的函數(shù),實現(xiàn)這樣的功能,把該字符串中的“#”號刪除,同時把后面連接的字符串前移。,12.2.3 指針的指針,【范例12-11】 使用指針的指針訪問字符串數(shù)組。01 #include 02
28、 int main(void)03 {04 char *array[]={"Winter","Spring","Summer","Fall"};05 char **p; /*指針的指針*/06 int i;07 for(i=0;i<4;i++)08 {09 p=array+i;
29、/*指針的指針p指向array+i所指向的字符串的首地址*/10 printf("%s\n",*p); /*輸出數(shù)組中的每一個字符串*/11 }12 return 0;13 },12.2.3 指針的指針,,,12.2.4 指針數(shù)組和數(shù)組指針,什么是指針數(shù)組呢?指針數(shù)組是指數(shù)組由指針類型的元素組成。比如:int *p[10]。這里數(shù)組p是由10個指向整型元素的指針組成的
30、,比如p[0]就是一個指針變量,它的使用跟一般的指針用法一樣,無非是這些指針有同一個名字需要使用下標來區(qū)分。例如我們有下面的定義:char *p[2];char array[2][20];p[0]=array[0];p[1]=array[1];如下圖所示。,,12.2.4 指針數(shù)組和數(shù)組指針,數(shù)組指針,比如:int (*p)[10]。指針p用來指向含有10個元素的整型數(shù)組?!痉独?2-12】 使用數(shù)組指針。01
31、#include 02 int main(void)03 {04 int array[2][3]={1,2,3,4,5,6};05 int i,j;06 int (*p)[3];07 p=array; /*p指向array下標為0那行的首地址*/08 for(i=0;i<2;i++)09 {10 for(j=0;j<3;j++)11
32、 printf("array[%d][%d]=%d\n",i,j,p[i][j]);12 }13 return 0;14 },12.3 綜合應用——報數(shù)游戲,【范例12-13】 有n個人圍成一圈,順序排號。從第一個人開始報數(shù)(從1到3報數(shù)),凡報到3的人退出圈子,問最后留下的那位是原來的第幾號。,12.5 跟我上機,1. 編寫C程序,使用指針,實現(xiàn)以下功能:編一個函數(shù)f
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
評論
0/150
提交評論