亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

iOS開發系列--C語言之指針

2019-11-14 20:15:22
字體:
來源:轉載
供稿:網友

概覽

指針是C語言的精髓,但是很多初學者往往對于指針的概念并不深刻,以至于學完之后隨著時間的推移越來越模糊,感覺指針難以掌握,本文通過簡單的例子試圖將指針解釋清楚,今天的重點有幾個方面:

  1. 什么是指針
  2. 數組和指針
  3. 函數指針

什么是指針

存放變量地址的變量我們稱之為“指針變量”,簡單的說變量p中存儲的是變量a的地址,那么p就可以稱為是指針變量,或者說p指向a。當我們訪問a變量的時候其實是程序先根據a取得a對應的地址,再到這個地址對應的存儲空間中拿到a的值,這種方式我們稱之為“直接引用”;而當我們通過p取得a的時候首先要先根據p轉換成p對應的存儲地址,再根據這個地址到其對應的存儲空間中拿到存儲內容,它的內容其實就是a的地址,然后根據這個地址到對應的存儲空間中取得對應的內容,這個內容就是a的值,這種通過p找到a對應地址再取值的方式成為“間接引用”。這里以表格形式列出a和p的存儲以幫助大家理解上面說的內容:

address1.1

接下來,看一下指針的賦值

////  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); //結果:address(a)=5fbff81c,address(p)=5fbff81c    printf("a=%d,p=%d/n",a,*p); //結果:a=1,p=1    *p=2;    printf("a=%d,*p=%d/n",a,*p); //結果:a=2,p=2        int b=8;    char c= 1;    int *q=&c;    printf("address(b)=%x,address(c)=%x/n",&b,&c);//結果:    printf("c=%d,q=%d/n", c, *q); //結果:c=1,q=2049,為什么q的值不是1呢?        return 0;}

需要說明兩點:

a.int *p;中的*只是表示p變量是一個指針變量;而打印*p的時候,*p中的*是操作符,表示p指針指向的變量的存儲空間(當前存儲就是1),同時我們也看到了*p==a;修改了*p也就是修改了p指向的存儲空間的內容,也就修改了a,所以第二次打印a=2;

b.指針所指向的類型必須和定義指針時聲明的類型相同;上面指針q定義成了int型而指向了char型,結果輸出*q打印出了2049,具體原因見下圖(假設在16位編譯器下,指針長度為2字節)

address1.2

由于局部變量是存儲在棧里面的,所以先存儲b再存儲a、p,當打印*p的時候,其實就是以p指向的地址對應的空間開始取兩個字節的數據(因為定義p的時候它指向的是int型,在16位編譯器下int類型的長度為2),剛好定義的b和c空間連續,所以就取到b的其中一個字節,最后*p二進制存儲為“0000100000000001”(見上圖黃色背景內容),十進制表示就是2049;

c.指針變量占用的空間和它所指向的變量類型無關,只跟編譯器位數有關(準確的說只跟尋址方式有關);

數組和指針

由于數組的存儲是連續的,數組名就是數組的地址,這樣一來數組和指針就有著很微妙的關系,先看以下例子:

////  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]; //等價于:*p=a;        printf("len=%lu/n",sizeof(int));//取得int長度為2        //指針加1代表地址向后挪動所指向類型的長度位(這里類型是int,長度為2)    //也就是說p指向a[0],p+1指向a[1],以此類推,所以我們通過指針也可以取出數組元素    for(int i=0;i<3;++i){        //printf("a[%d]=%d/n",i,a[i]);        printf("a[%d]=%d/n",i,*(p+i));//由于a就代表數組的地址,其實這里還可以寫成*(a+i),但是注意這里*(p+i)可以寫成*(p++),但是*(a+i)不能寫成*(a++),因為數組名是常量    }    /*輸出結果:     a[0]=1     a[1]=2     a[2]=3     */             changeValue(p); //等價于:changeValue(a)    for(int i=0;i<3;++i){        printf("a[%d]=%d/n",i,a[i]);    }    /*輸出結果:     a[0]=2     a[1]=2     a[2]=3     */        changeValue2(a); //等價于:changeValue2(p)    for(int i=0;i<3;++i){        printf("a[%d]=%d/n",i,a[i]);    }    /*輸出結果:     a[0]=3     a[1]=2     a[2]=3     */        return 0;}

從上面的例子我們可以得出如下結論:

  1. 數組名a==&a[0]==*p;
  2. 如果p指向一個數組,那么p+1指向數組的下一個元素,同時注意p+1移動的長度并不固定,具體需要根據p指向的數據類型而定;
  3. 指針可以寫成p++形式,但是數組名不可以,因為數組名是常量
  4. 不管函數的形參為數組還是指針,實參都可以使用數組名或指針;

擴展--字符串和指針

由于在C語言中字符串就是字符數組,下面不妨看一下字符串和數組的關系:

////  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);//結果:5fbff820,Kenshin,同一個變量a是輸出字符串還是輸出地址,根據格式參數而定    printf(a); //結果:Kenshin    printf("/n");        char b[]="Kenshin";    char *p=b;    printf("b=%s,p=%s/n",b,p);//結果:b=Kenshin,p=Kenshin        //指針存儲的是地址,而數組名存儲的也是地址,既然字符數組可以表示字符串,那么指向字符的指針同樣也可以,如下方式可以更簡單的定義一個字符串    char *c="Kenshin"; //等價于char c[]="Kenshin";    printf("c=%s/n",c); //結果:c=Kenshin        return 0;}

以上代碼中注釋基本已經很清楚了,這里需要指出是為什么printf(a)能夠直接輸出字符串呢?

我們看一下printf()的定義:int     printf(const char * __restrict, ...) __printflike(1, 2);

其實printf的參數要求是指向字符類型的指針,而結合上面的例子和我們之前說的,如果函數形參是指針類型那么可以傳入函數名,因此也就能正確輸出字符串的內容了。類似的還有上一篇文章中說的strcat()、strcpy()等函數均是如此。

函數指針

在弄清函數指針的問題之前,我們不妨先來看一下返回指針類型數據的函數,畢竟指針類型也是C語言的數據類型,下面以一個字符串轉換為大寫字符的程序為例,在這個例子中不僅可以看到返回值為指針類型的函數同時還可以看到前面說到的指針移動操作:

////  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; //保留最初地址,因為后面的循環會改變字符串最初地址    int len='a'-'A'; //大小寫ASCII碼差值相等    while (*a!='/0') { //字符是否結束        if(*a>'a'&&*a<'z'){//如果是小寫字符            *(a++) -= len; //*a表示數組對應的字符(-32變為小寫),a++代表移動到下一個字符        }    }       return b;}int main(int argc, const char * argv[]) {    char a[]="hello";    char *p=toUpper(a);    printf("%s/n",p); //結果:HELLO    return 0;}

大家都是知道函數只能有一個返回值,如果需要返回多個值,怎么辦,其實很簡單,只要將指針作為函數參數傳遞就可以了,在下面的例子中我們再次看到指針作為參數進行傳遞。

////  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);//結果:a+b=3,a-b=-1    return 0;}

函數也是在內存中存儲的,當然函數也有一個起始地址(事實上函數名就是函數的起始地址),這里同樣需要弄清函數指針的關系。函數指針定義的形式:返回值類型 (*指針變量名)(形參1,形參2),拿到函數指針其實我們就相當于拿到了這個函數,函數的操作都可以通過指針來完成,而且通過前面的例子可以看到指針作為C語言的數據類型,可以作為參數、作為返回值,那么當然函數指針同樣可以作為函數的參數和返回值:

////  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;}//函數指針作為參數進行傳遞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;//函數名就是函數首地址,等價于:int (*p)(int,int);p=sum;    int c=p(a,b);    printf("a+b=%d/n",c); //結果:a+b=3            //函數作為參數傳遞    printf("%d/n",operate(a, b, sum)); //結果:3    printf("%d/n",operate(a, b, sub)); //結果:-1        return 0;}

函數指針可以作為函數參數進行傳遞,實在太強大了,是不是想起了C#中的委托?記得C#書籍中經常提到委托類似于函數指針,其實說的就是上面的情況。需要注意的是,普通的指針可以寫成p++進行移動,而函數指針寫成p++并沒有意義。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲国产三级网| 岛国av一区二区| 国产91色在线|| 黑人巨大精品欧美一区二区一视频| 91精品国产自产在线观看永久| 欧美激情国内偷拍| 日韩麻豆第一页| 欧美福利视频在线观看| 亚洲精品suv精品一区二区| 91最新在线免费观看| 久久天天躁狠狠躁夜夜躁2014| 爱福利视频一区| 日韩毛片在线看| 按摩亚洲人久久| 亚洲国产精品成人va在线观看| 日韩欧美一区二区在线| 亚洲精品美女久久久久| 欧美精品免费在线| 日韩欧美中文在线| 黑人精品xxx一区一二区| 亚洲电影成人av99爱色| 日韩中文在线中文网在线观看| 午夜精品一区二区三区在线| 日韩美女在线播放| 国产精品男人爽免费视频1| 日韩欧美极品在线观看| 日韩欧美在线视频免费观看| 久久久久久亚洲精品中文字幕| 一区二区在线视频播放| 国产+成+人+亚洲欧洲| 久久久91精品国产一区不卡| 日韩欧美亚洲范冰冰与中字| 久久久亚洲天堂| 免费99精品国产自在在线| 色综合久久88色综合天天看泰| 久热精品视频在线| 91精品在线观看视频| 中文字幕精品在线视频| 亚洲午夜性刺激影院| 97超碰色婷婷| 国产成人激情视频| 久久久久中文字幕| 中文字幕无线精品亚洲乱码一区| 日本精品免费一区二区三区| 国产精品日韩电影| 亚洲精品久久视频| 久久久久久亚洲精品中文字幕| 欧美日韩亚洲国产一区| 国产91对白在线播放| 国产精品jizz在线观看麻豆| 亚洲欧美另类自拍| 欧美电影免费看| 精品国产乱码久久久久酒店| 国产欧美精品va在线观看| 日韩成人小视频| 日韩av在线一区二区| 国产午夜一区二区| 91日韩在线视频| 中文字幕av一区二区三区谷原希美| 欧美成人免费大片| 日韩中文视频免费在线观看| 在线丨暗呦小u女国产精品| 国产精品激情av电影在线观看| 成人在线免费观看视视频| 尤物tv国产一区| 久久精品99久久久香蕉| 国产精品免费一区二区三区都可以| 亚洲自拍偷拍视频| 日韩一区二区三区xxxx| 国产精品久久久久久五月尺| 日韩av黄色在线观看| 日韩高清av在线| 久久久久久久一区二区| 久久99国产精品自在自在app| 少妇精69xxtheporn| 亚洲美腿欧美激情另类| 国产成人精品日本亚洲| 日韩av电影在线免费播放| 亚洲天堂男人的天堂| 久久久久久国产精品三级玉女聊斋| 亚洲精品久久久久久久久久久久久| www.亚洲一二| 97视频免费在线看| 亚洲国内精品视频| 日韩国产欧美区| 亚洲自拍偷拍色图| 精品久久久久久久久久久| 国产精品久久久久久久电影| 日本久久91av| 日韩风俗一区 二区| 欧美成人国产va精品日本一级| 亚洲xxxx做受欧美| 日日骚av一区| 久久久久久欧美| 亚洲人成在线观看| 亚洲在线免费观看| 欧美成人激情视频免费观看| 亚洲а∨天堂久久精品9966| 最近中文字幕mv在线一区二区三区四区| 精品亚洲一区二区| 欧美亚洲在线视频| 日本久久久久久久| 色吧影院999| 日韩免费在线看| 久久精品国产成人精品| 日韩精品中文字幕在线观看| 欧美激情久久久| 亚洲欧美制服中文字幕| 亚洲精品一区在线观看香蕉| 日韩精品久久久久| 国产精品视频网站| 久久精品视频免费播放| 夜夜躁日日躁狠狠久久88av| 91麻豆桃色免费看| 日韩欧美在线免费| 91精品免费久久久久久久久| 97视频在线看| 97久久伊人激情网| 亚洲免费视频在线观看| 色偷偷亚洲男人天堂| 国产日韩精品一区二区| 国产成人中文字幕| 日韩av一区二区在线观看| 隔壁老王国产在线精品| 色偷偷噜噜噜亚洲男人| 欧美激情中文字幕在线| 精品亚洲永久免费精品| 国产精品视频免费观看www| 日本久久久久亚洲中字幕| 色综合影院在线| 欧美激情视频网站| 久久久久久久久综合| 日韩电视剧在线观看免费网站| 亚洲国产成人在线播放| 欧洲日韩成人av| 亚洲伊人一本大道中文字幕| 欧美一区视频在线| 亚洲欧美日韩天堂一区二区| 日韩中文字幕在线播放| 久久影视电视剧免费网站清宫辞电视| 欧美大成色www永久网站婷| 国产美女精品视频免费观看| 久久精品国产亚洲精品| 精品久久久精品| 久久6精品影院| 亚洲日韩中文字幕| 一区二区三区 在线观看视| 欧美老女人bb| 欧美乱大交做爰xxxⅹ性3| 国产精品视频久久久久| 亚洲一区二区三区视频| 成人激情电影一区二区| 欧美另类高清videos| 91精品国产免费久久久久久| 日产精品99久久久久久| 中文字幕亚洲自拍| 国产欧美婷婷中文| 亚洲小视频在线观看| 一区二区三区动漫| 日本精品久久久久影院| 国产亚洲精品91在线| 亚洲缚视频在线观看| 欧美精品videos| 57pao国产精品一区|