IOS開發(fā)之路--C語(yǔ)言指針
概覽
指針是C語(yǔ)言的精髓,但是很多初學(xué)者往往對(duì)于指針的概念并不深刻,以至于學(xué)完之后隨著時(shí)間的推移越來(lái)越模糊,感覺指針難以掌握,本文通過(guò)簡(jiǎn)單的例子試圖將指針解釋清楚,今天的重點(diǎn)有幾個(gè)方面:
什么是指針 數(shù)組和指針 函數(shù)指針
什么是指針
存放變量地址的變量我們稱之為“指針變量”,簡(jiǎn)單的說(shuō)變量p中存儲(chǔ)的是變量a的地址,那么p就可以稱為是指針變量,或者說(shuō)p指向a。當(dāng)我們?cè)L問(wèn)a變量的時(shí)候其實(shí)是程序先根據(jù)a取得a對(duì)應(yīng)的地址,再到這個(gè)地址對(duì)應(yīng)的存儲(chǔ)空間中拿到a的值,這種方式我們稱之為“直接引用”;而當(dāng)我們通過(guò)p取得a的時(shí)候首先要先根據(jù)p轉(zhuǎn)換成p對(duì)應(yīng)的存儲(chǔ)地址,再根據(jù)這個(gè)地址到其對(duì)應(yīng)的存儲(chǔ)空間中拿到存儲(chǔ)內(nèi)容,它的內(nèi)容其實(shí)就是a的地址,然后根據(jù)這個(gè)地址到對(duì)應(yīng)的存儲(chǔ)空間中取得對(duì)應(yīng)的內(nèi)容,這個(gè)內(nèi)容就是a的值,這種通過(guò)p找到a對(duì)應(yīng)地址再取值的方式成為“間接引用”。這里以表格形式列出a和p的存儲(chǔ)以幫助大家理解上面說(shuō)的內(nèi)容:
接下來(lái),看一下指針的賦值
// // main.c // Point // // Created by Kenshin Cui on 14-7-05. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> int main(int argc, const char * argv[]) { int a=1; int *p; p=&a; //也可以直接給指針變量賦值:int *p=&a; printf("address(a)=%x,address(p)=%x\n",&a,p); //結(jié)果:address(a)=5fbff81c,address(p)=5fbff81c printf("a=%d,p=%d\n",a,*p); //結(jié)果:a=1,p=1 *p=2; printf("a=%d,*p=%d\n",a,*p); //結(jié)果:a=2,p=2 int b=8; char c= 1; int *q=&c; printf("address(b)=%x,address(c)=%x\n",&b,&c);//結(jié)果: printf("c=%d,q=%d\n", c, *q); //結(jié)果:c=1,q=2049,為什么q的值不是1呢? return 0; }
需要說(shuō)明兩點(diǎn):
a.int *p;中的*只是表示p變量是一個(gè)指針變量;而打印*p的時(shí)候,*p中的*是操作符,表示p指針指向的變量的存儲(chǔ)空間(當(dāng)前存儲(chǔ)就是1),同時(shí)我們也看到了*p==a;修改了*p也就是修改了p指向的存儲(chǔ)空間的內(nèi)容,也就修改了a,所以第二次打印a=2;
b.指針?biāo)赶虻念愋捅仨毢投x指針時(shí)聲明的類型相同;上面指針q定義成了int型而指向了char型,結(jié)果輸出*q打印出了2049,具體原因見下圖(假設(shè)在16位編譯器下,指針長(zhǎng)度為2字節(jié))
由于局部變量是存儲(chǔ)在棧里面的,所以先存儲(chǔ)b再存儲(chǔ)a、p,當(dāng)打印*p的時(shí)候,其實(shí)就是以p指向的地址對(duì)應(yīng)的空間開始取兩個(gè)字節(jié)的數(shù)據(jù)(因?yàn)槎xp的時(shí)候它指向的是int型,在16位編譯器下int類型的長(zhǎng)度為2),剛好定義的b和c空間連續(xù),所以就取到b的其中一個(gè)字節(jié),最后*p二進(jìn)制存儲(chǔ)為“0000100000000001”(見上圖黃色背景內(nèi)容),十進(jìn)制表示就是2049;
c.指針變量占用的空間和它所指向的變量類型無(wú)關(guān),只跟編譯器位數(shù)有關(guān)(準(zhǔn)確的說(shuō)只跟尋址方式有關(guān));
數(shù)組和指針
由于數(shù)組的存儲(chǔ)是連續(xù)的,數(shù)組名就是數(shù)組的地址,這樣一來(lái)數(shù)組和指針就有著很微妙的關(guān)系,先看以下例子:
// // main.c // Point // // Created by Kenshin Cui on 14-7-05. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> void changeValue(int a[]){ a[0]=2; } void changeValue2(int *p){ p[0]=3; } int main(int argc, const char * argv[]) { int a[]={1,2,3}; int *p=&a[0]; //等價(jià)于:*p=a; printf("len=%lu\n",sizeof(int));//取得int長(zhǎng)度為2 //指針加1代表地址向后挪動(dòng)所指向類型的長(zhǎng)度位(這里類型是int,長(zhǎng)度為2) //也就是說(shuō)p指向a[0],p+1指向a[1],以此類推,所以我們通過(guò)指針也可以取出數(shù)組元素 for(int i=0;i<3;++i){ //printf("a[%d]=%d\n",i,a[i]); printf("a[%d]=%d\n",i,*(p+i));//由于a就代表數(shù)組的地址,其實(shí)這里還可以寫成*(a+i),但是注意這里*(p+i)可以寫成*(p++),但是*(a+i)不能寫成*(a++),因?yàn)閿?shù)組名是常量 } /*輸出結(jié)果: a[0]=1 a[1]=2 a[2]=3 */ changeValue(p); //等價(jià)于:changeValue(a) for(int i=0;i<3;++i){ printf("a[%d]=%d\n",i,a[i]); } /*輸出結(jié)果: a[0]=2 a[1]=2 a[2]=3 */ changeValue2(a); //等價(jià)于:changeValue2(p) for(int i=0;i<3;++i){ printf("a[%d]=%d\n",i,a[i]); } /*輸出結(jié)果: a[0]=3 a[1]=2 a[2]=3 */ return 0; }
從上面的例子我們可以得出如下結(jié)論:
數(shù)組名a==&a[0]==*p; 如果p指向一個(gè)數(shù)組,那么p+1指向數(shù)組的下一個(gè)元素,同時(shí)注意p+1移動(dòng)的長(zhǎng)度并不固定,具體需要根據(jù)p指向的數(shù)據(jù)類型而定; 指針可以寫成p++形式,但是數(shù)組名不可以,因?yàn)閿?shù)組名是常量 不管函數(shù)的形參為數(shù)組還是指針,實(shí)參都可以使用數(shù)組名或指針;擴(kuò)展--字符串和指針
由于在C語(yǔ)言中字符串就是字符數(shù)組,下面不妨看一下字符串和數(shù)組的關(guān)系:
// // main.c // Point // // Created by Kenshin on 14-7-05. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> int main(int argc, const char * argv[]) { char a[]="Kenshin"; printf("%x,%s\n",a,a);//結(jié)果:5fbff820,Kenshin,同一個(gè)變量a是輸出字符串還是輸出地址,根據(jù)格式參數(shù)而定 printf(a); //結(jié)果:Kenshin printf("\n"); char b[]="Kenshin"; char *p=b; printf("b=%s,p=%s\n",b,p);//結(jié)果:b=Kenshin,p=Kenshin //指針存儲(chǔ)的是地址,而數(shù)組名存儲(chǔ)的也是地址,既然字符數(shù)組可以表示字符串,那么指向字符的指針同樣也可以,如下方式可以更簡(jiǎn)單的定義一個(gè)字符串 char *c="Kenshin"; //等價(jià)于char c[]="Kenshin"; printf("c=%s\n",c); //結(jié)果:c=Kenshin return 0; }
以上代碼中注釋基本已經(jīng)很清楚了,這里需要指出是為什么printf(a)能夠直接輸出字符串呢?
我們看一下printf()的定義:int printf(const char * __restrict, ...) __printflike(1, 2);
其實(shí)printf的參數(shù)要求是指向字符類型的指針,而結(jié)合上面的例子和我們之前說(shuō)的,如果函數(shù)形參是指針類型那么可以傳入函數(shù)名,因此也就能正確輸出字符串的內(nèi)容了。類似的還有上一篇文章中說(shuō)的strcat()、strcpy()等函數(shù)均是如此。
函數(shù)指針
在弄清函數(shù)指針的問(wèn)題之前,我們不妨先來(lái)看一下返回指針類型數(shù)據(jù)的函數(shù),畢竟指針類型也是C語(yǔ)言的數(shù)據(jù)類型,下面以一個(gè)字符串轉(zhuǎn)換為大寫字符的程序?yàn)槔?,在這個(gè)例子中不僅可以看到返回值為指針類型的函數(shù)同時(shí)還可以看到前面說(shuō)到的指針移動(dòng)操作:
// // main.c // Point // // Created by Kenshin Cui on 14-06-28. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> char * toUpper(char *a){ char *b=a; //保留最初地址,因?yàn)楹竺娴难h(huán)會(huì)改變字符串最初地址 int len='a'-'A'; //大小寫ASCII碼差值相等 while (*a!='\0') { //字符是否結(jié)束 if(*a>'a'&&*a<'z'){//如果是小寫字符 *(a++) -= len; //*a表示數(shù)組對(duì)應(yīng)的字符(-32變?yōu)樾懀琣++代表移動(dòng)到下一個(gè)字符 } } return b; } int main(int argc, const char * argv[]) { char a[]="hello"; char *p=toUpper(a); printf("%s\n",p); //結(jié)果:HELLO return 0; }
大家都是知道函數(shù)只能有一個(gè)返回值,如果需要返回多個(gè)值,怎么辦,其實(shí)很簡(jiǎn)單,只要將指針作為函數(shù)參數(shù)傳遞就可以了,在下面的例子中我們?cè)俅慰吹街羔樧鳛閰?shù)進(jìn)行傳遞。
// // main.c // Point // // Created by Kenshin Cui on 14-6-28. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> int operate(int a,int b,int *c){ *c=a-b; return a+b; } int main(int argc, const char * argv[]) { int a=1,b=2,c,d; d=operate(a, b, &c); printf("a+b=%d,a-b=%d\n",d,c);//結(jié)果:a+b=3,a-b=-1 return 0; }
函數(shù)也是在內(nèi)存中存儲(chǔ)的,當(dāng)然函數(shù)也有一個(gè)起始地址(事實(shí)上函數(shù)名就是函數(shù)的起始地址),這里同樣需要弄清函數(shù)指針的關(guān)系。函數(shù)指針定義的形式:返回值類型 (*指針變量名)(形參1,形參2),拿到函數(shù)指針其實(shí)我們就相當(dāng)于拿到了這個(gè)函數(shù),函數(shù)的操作都可以通過(guò)指針來(lái)完成,而且通過(guò)前面的例子可以看到指針作為C語(yǔ)言的數(shù)據(jù)類型,可以作為參數(shù)、作為返回值,那么當(dāng)然函數(shù)指針同樣可以作為函數(shù)的參數(shù)和返回值:
// // main.c // Point // // Created by Kenshin Cui on 14-6-28. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> int sum(int a,int b){ return a+b; } int sub(int a,int b){ return a-b; } //函數(shù)指針作為參數(shù)進(jìn)行傳遞 int operate(int a,int b,int (*p)(int,int)){ return p(a,b); } int main(int argc, const char * argv[]) { int a=1,b=2; int (*p)(int ,int)=sum;//函數(shù)名就是函數(shù)首地址,等價(jià)于:int (*p)(int,int);p=sum; int c=p(a,b); printf("a+b=%d\n",c); //結(jié)果:a+b=3 //函數(shù)作為參數(shù)傳遞 printf("%d\n",operate(a, b, sum)); //結(jié)果:3 printf("%d\n",operate(a, b, sub)); //結(jié)果:-1 return 0; }
函數(shù)指針可以作為函數(shù)參數(shù)進(jìn)行傳遞,實(shí)在太強(qiáng)大了,是不是想起了C#中的委托?記得C#書籍中經(jīng)常提到委托類似于函數(shù)指針,其實(shí)說(shuō)的就是上面的情況。需要注意的是,普通的指針可以寫成p++進(jìn)行移動(dòng),而函數(shù)指針寫成p++并沒有意義。
相關(guān)文章
全面解析iOS中同步請(qǐng)求、異步請(qǐng)求、GET請(qǐng)求、POST請(qǐng)求
通過(guò)本文給大家全面解析了iOS中同步請(qǐng)求、異步請(qǐng)求、GET請(qǐng)求、POST請(qǐng)求,非常不錯(cuò),具有參考借鑒價(jià)值,感興趣的朋友一起學(xué)習(xí)吧2016-08-08實(shí)例講解iOS中的CATransition轉(zhuǎn)場(chǎng)動(dòng)畫使用
CATransition類為應(yīng)用程序的轉(zhuǎn)場(chǎng)動(dòng)畫提供了很多可控制參數(shù),接下來(lái)我們就以幾個(gè)實(shí)例講解iOS中的CATransition轉(zhuǎn)場(chǎng)動(dòng)畫使用,需要的朋友可以參考下2016-06-06IOS網(wǎng)絡(luò)請(qǐng)求之NSURLSession使用詳解
這篇文章主要介紹了IOS網(wǎng)絡(luò)請(qǐng)求之NSURLSession使用詳解,今天使用NSURLConnection分別實(shí)現(xiàn)了get、post、表單提交、文件上傳、文件下載,有興趣的可以了解一下。2017-02-02Xcode8以及iOS10適配等常見問(wèn)題匯總(整理篇)
隨著iOS 10的更新以及Xcdoe 8的更新出現(xiàn)了很多問(wèn)題,今天小編抽時(shí)間給大家整理下我遇到的坑特此分享到腳本之家平臺(tái),供大家參考2016-09-09iOS中的導(dǎo)航欄UINavigationBar與工具欄UIToolBar要點(diǎn)解析
UINavigation可以附著于導(dǎo)航控制器之中使用,也可以在controller中單獨(dú)使用,這里我們將來(lái)看iOS中的導(dǎo)航欄UINavigationBar與工具欄UIToolBar要點(diǎn)解析.2016-06-06iOS中UIAlertController設(shè)置自定義標(biāo)題與內(nèi)容的方法
UIAlertController是iOS8推出的新概念,取代了之前的 UIAlertView和UIActionSheet(雖然現(xiàn)在仍可以使用,但是會(huì)有警告)。下面這篇文章主要給大家介紹了關(guān)于iOS中UIAlertController如何設(shè)置自定義標(biāo)題與內(nèi)容的相關(guān)資料,需要的朋友可以參考下。2017-10-10舉例講解iOS開發(fā)中拖動(dòng)視圖的實(shí)現(xiàn)
這篇文章主要介紹了舉例講解iOS開發(fā)中的拖動(dòng)視圖實(shí)現(xiàn),代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-10-10iOS-Mac配置Tomcat教程 Mac環(huán)境配置Tomcat教程
這篇文章主要介紹了iOS-Mac配置Tomcat教程,Mac環(huán)境配置Tomcat,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11iOS如何用100行代碼實(shí)現(xiàn)簡(jiǎn)單的抽屜效果
最近在網(wǎng)上看到一些抽屜效果,看起來(lái)很酷!很眩!但是,下不下來(lái)看代碼, 所以決定還是自己寫吧!!這篇文章通過(guò)近100行的代碼就實(shí)現(xiàn)了簡(jiǎn)單的抽屜效果,有需要的朋友們可以參考借鑒,下面來(lái)一起看看吧。2016-10-10iOS開發(fā)之TextField禁用粘貼、選擇和全選功能
這篇文章主要為大家詳細(xì)介紹了iOS開發(fā)之TextField禁用粘貼、選擇和全選功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09