C語(yǔ)言的那些小秘密之指針(一)

發(fā)布時(shí)間:2016-2-18 14:13    發(fā)布者:designapp
關(guān)鍵詞: C語(yǔ)言 , 指針
  懂得C語(yǔ)言的人都知道,C語(yǔ)言之所以強大,以及其自由性,絕大部分體現在其靈活的指針運用上。因此,說(shuō)指針是c語(yǔ)言的靈魂,一點(diǎn)都不為過(guò)。所以從我的標題加了個(gè)(一)也可以看出指針的重要性,我盡可能的向大家交代清楚我對于指針的理解。所以在講解的過(guò)程中我盡可能的用代碼加文字的描述方式,通過(guò)代碼的分析來(lái)加深我們對于指針的理解,我給出的都是完整的代碼,所以讀者可以在看的過(guò)程中直接copy下去即可運行,希望下面的講解能夠對你有所幫助。
  首先讓我們來(lái)看看定義一個(gè)指針的一般形式為:
  基類(lèi)型 *指針變量名
  看了上面的指針的定義形式,我們可能對于有些地方會(huì )有疑惑,如為什么要指定基類(lèi)型呢?因為我們都知道整型和字符型在內存中占的字節數是不相同的,當我們進(jìn)行指針的移動(dòng)和指針的運算時(shí),如果指針指向的是一個(gè)整型變量,那么指針移動(dòng)一個(gè)位置就是移動(dòng)4個(gè)字節,但是如果指針指向的是一個(gè)字符型的變量,那么指針移動(dòng)的就是一個(gè)字節,因此我們必須規定指針變量所指向的基類(lèi)型。
  為了不枯燥的講解我們來(lái)看看下面的代碼吧。(注意:本博客的所有代碼均使用vc6編譯運行,所以可能有的規則跟C語(yǔ)言的稍有區別)
  #include
  int main()
  {
  int a,b;
  int *pointer_1,*pointer_2;
  a=100;
  b=200;
  pointer_1=&a;
  pointer_2=&b;
  printf("--------------------變換前-------------------\n");
  printf("a=%d\tb=%d\n",a,b);
  printf("*pointer_1=%d\t*pointer_2=%d\n",*pointer_1,*pointer_2);
  *pointer_1=300;
  int c=500;
  pointer_2=&c;
  printf("--------------------變換后-------------------\n");
  printf("a=%d\t*pointer_1=%d\n",a,*pointer_1);
  printf("c=%d\tb=%d\t*pointer_2=%d\n",c,b,*pointer_2);
  }
  運行結果如下:
  


  在此我們定義了兩個(gè)整型指針int *pointer_1,*pointer_2;,它們分別指向變量a和b,值得注意的是*pointer_1和a、*pointer_2和b是共用同一個(gè)存儲空間的,當我們在接下類(lèi)的代碼中改變 *pointer_1=300;時(shí),由輸出就可以看出來(lái)a的值也跟隨發(fā)生了改變。但是當我們聲明了一個(gè) int c=500;之后,使用pointer_2=&c;,b的值不變,僅僅是改變*pointer_2,因為我僅僅是改變了*pointer_2指向了c的存儲空間,如果有有興趣的讀者可以自己驗證下如果我們修改了a的值之后*pointer_1的值會(huì )跟隨一起改變,因為他們指向的是同一個(gè)存儲空間。
  接下來(lái)看看如何在函數的參數中來(lái)使用指針。
  #include
  swap(int p1,int p2)
  {
  int temp;
  temp=p1;
  p1=p2;
  p2=temp;
  }
  int main()
  {
  int a,b;
  int *pointer_1,*pointer_2;
  int c,d;
  c=a;
  d=b;
  pointer_1=&a;
  pointer_2=&b;
  a=20;
  b=30;
  swap(a,b);
  printf("a=%d\tb=%d\n",a,b);
  printf("a=%d\tb=%d\n",*pointer_1,*pointer_2);
  printf("c=%d\td=%d\n",c,d);
  }
  初步分析上面的代碼,看似是要通過(guò)一個(gè)函數的調用來(lái)實(shí)現一個(gè)a、b的交換,還有就是通過(guò)c=a;、 d=b;來(lái)實(shí)現對c、d賦初值。先來(lái)看看下面的運行結果:
  


  結果跟我們想象的不一樣,a、b沒(méi)有實(shí)現交換的原因是因為我們使用的是傳值,而不是傳址,所以調用的過(guò)程中做的處理就是把a、b的值復制到另外申請的兩個(gè)空間p1、p2中去,因而交換操作是在p1、p2的空間中進(jìn)行的,所以對于a、b的值并沒(méi)有影響。c、d的初值為什么沒(méi)有跟a、b的值一樣呢,因為我們在初始化的過(guò)程中給c、d賦初值的時(shí)候a、b的并沒(méi)有給定初值,所以a、b的初值是在編譯的過(guò)程中由系統給定的,又因為我們申請的c、d的空間是跟a、b沒(méi)有任何關(guān)系的,所以接下來(lái)再對a、b賦初值的時(shí)候c、d的初值并不會(huì )改變。
  下一個(gè)代碼:
  #include
  swap(int *p1,int *p2)
  {
  int *temp;
  temp=p1;
  p1=p2;
  p2=temp;
  }
  int main()
  {
  int a,b;
  int *pointer_1,*pointer_2;
  int c,d;
  c=a;
  d=b;
  pointer_1=&a;
  pointer_2=&b;
  a=20;
  b=30;
  printf("********************調用前******************\n");
  printf("a=%d\tb=%d\n",a,b);
  swap(pointer_1,pointer_2);
  printf("********************調用后******************\n");
  printf("a=%d\tb=%d\n",a,b);
  printf("*pointer_1=%d\t*pointer_2=%d\n",*pointer_1,*pointer_2);
  printf("c=%d\td=%d\n",c,d);
  return 0;
  }
  看看上面這個(gè)代碼似乎滿(mǎn)足了我們前面說(shuō)的傳址的要求,那先讓我們來(lái)看看實(shí)驗結果吧。
  


  結果似乎也是出乎我們的意料之外,為什么使用了傳值卻還是沒(méi)有能夠實(shí)現呢?如果我們在調用函數中加上一句 printf("*p1=%d\t*p2=%d\n",*p1,*p2);,得到下面的結果:


  從結果來(lái)看似乎告訴我們,我們已經(jīng)實(shí)現交換了,但是為什么沒(méi)有能夠返回來(lái)呢?在這里要注意了,因為我們在函數的交換語(yǔ)句僅僅是改變了局部指針變量p1和p2的值,所以沒(méi)有改變a、b的值,所以使用printf("*p1=%d\t*p2=%d\n",*p1,*p2);使得我們的確看到了a、b交換的假象,僅僅是改變了局部變量p1和p2的值。
  下一個(gè)代碼:
  #include
  #include
  swap(int *p1,int *p2)
  {
  int *temp;
  temp=(int *)malloc(sizeof(int));
  *temp=*p1;
  *p1=*p2;
  *p2=*temp;
  free(temp);
  }
  int main()
  {
  int a,b;
  int *pointer_1,*pointer_2;
  int c,d;
  c=a;
  d=b;
  pointer_1=&a;
  pointer_2=&b;
  a=20;
  b=30;
  printf("********************調用前******************\n");
  printf("a=%d\tb=%d\n",a,b);
  swap(pointer_1,pointer_2);
  printf("********************調用后******************\n");
  printf("a=%d\tb=%d\n",a,b);
  printf("*pointer_1=%d\t*pointer_2=%d\n",*pointer_1,*pointer_2);
  printf("c=%d\td=%d\n",c,d);
  return 0;
  }
  看看也行結果:
  


  最后終于出現了一個(gè)我們想要的結果了。從以上的分析讀者自己也知道原因所在了吧,這里操作的才是p1、p2所指向的地址,才真正的做到了對于a、b存儲空間的數值的交換。細心的讀者可能看到了我們在代碼中用了紅色部分標記的代碼,它完全可以用一句int temp;來(lái)替代,之所以我們在這里要用int *temp;無(wú)非是要大家牢記對于指針一些特殊的使用,如果我們沒(méi)有這句temp=(int *)malloc(sizeof(int));,以上代碼在編譯的過(guò)程中是不會(huì )有任何錯誤的,但是在運行的過(guò)程中就會(huì )出現錯誤,所以通常情況下我們在使用指針的過(guò)程中,要特別注意野指針情況的出現,以免出現一些莫名奇妙的有錯。
                               
               
本文地址:http://selenalain.com/thread-160866-1-1.html     【打印本頁(yè)】

本站部分文章為轉載或網(wǎng)友發(fā)布,目的在于傳遞和分享信息,并不代表本網(wǎng)贊同其觀(guān)點(diǎn)和對其真實(shí)性負責;文章版權歸原作者及原出處所有,如涉及作品內容、版權和其它問(wèn)題,我們將根據著(zhù)作權人的要求,第一時(shí)間更正或刪除。
您需要登錄后才可以發(fā)表評論 登錄 | 立即注冊

關(guān)于我們  -  服務(wù)條款  -  使用指南  -  站點(diǎn)地圖  -  友情鏈接  -  聯(lián)系我們
電子工程網(wǎng) © 版權所有   京ICP備16069177號 | 京公網(wǎng)安備11010502021702
快速回復 返回頂部 返回列表
午夜高清国产拍精品福利|亚洲色精品88色婷婷七月丁香|91久久精品无码一区|99久久国语露脸精品|动漫卡通亚洲综合专区48页