• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            幽魂國度

             

            指針的指針

            指向另一指針的指針:轉自風過無痕博客
             

            一. 回顧指針概念:
            早在本系列第二篇中我就對指針的實質進行了闡述。今天我們又要學習一個叫做指向另一指針地址的指針。讓我們先回顧一下指針的概念吧!
            當我們程序如下申明變量:
            short int i;
            char a;
            short int * pi;
            程序會在內存某地址空間上為各變量開辟空間,如下圖所示。
            內存地址→6     7      8     9     10     11    12    13     14    15
            -------------------------------------------------------------------------------------
            …  |     |      |      |      |      |       |      |      |      |  
            -------------------------------------------------------------------------------------
                |short int i |char a|      |short int * pi|
            圖中所示中可看出:
            i 變量在內存地址5的位置,占兩個字節。
            a變量在內存地址7的位置,占一個字節。
            pi變量在內存地址9的位置,占兩個字節。(注:pi 是指針,我這里指針的寬度只有兩個字節,32位系統是四個字節)
            接下來如下賦值:
            i=50;
            pi=&i;
            經過上在兩句的賦值,變量的內存映象如下:
            內存地址→6     7      8     9     10     11    12    13      14     15
            --------------------------------------------------------------------------------------
            …  |    50      |      |      |    6         |      |      |       |  
            --------------------------------------------------------------------------------------
                |short int i |char a|      |short int * pi|
            看到沒有:短整型指針變量pi的值為6,它就是I變量的內存起始地址。所以,這時當我們對*pi進行讀寫操作時,其實就是對i變量的讀寫操作。如:
            *pi=5;   //就是等價于I=5;

            二. 指針的地址與指向另一指針地址的指針
            在上一節中,我們看到,指針變量本身與其它變量一樣也是在某個內存地址中的,如pi的內存起始地址是10。同樣的,我們也可能讓某個指針指向這個地址。
            看下面代碼:
            short int * * ppi;    //這是一個指向指針的指針,注意有兩個*號
            ppi=pi
            第一句:short int * * ppi;——申明了一個指針變量ppi,這個ppi是用來存儲(或稱指向)一個short int * 類型指針變量的地址。
            第二句:&pi那就是取pi的地址,ppi=pi就是把pi的地址賦給了ppi。即將地址值10賦值給ppi。如下圖:
            內存地址→6     7      8     9     10     11    12    13       14    15
            ------------------------------------------------------------------------------------
            …  |    50     |      |      |      6       |       10      |      |  
            ------------------------------------------------------------------------------------
                |short int i|char a|      |short int * pi|short int ** ppi|
            從圖中看出,指針變量ppi的內容就是指針變量pi的起始地址。于是……
            ppi的值是多少呢?——10。
            *ppi的值是多少呢?——6,即pi的值。
            **ppi的值是多少呢?——50,即I的值,也是*pi的值。
            呵呵!不用我說太多了,我相信你應明白這種指針了吧!

            三. 一個應用實例
            1. 設計一個函數:void find1(char array[], char search, char * pi)
            要求:這個函數參數中的數組array是以0值為結束的字符串,要求在字符串array中查找字符是參數search里的字符。如果找到,函數通過第三個參數(pa)返回值為array字符串中第一個找到的字符的地址。如果沒找到,則為pa為0。
            設計:依題意,實現代碼如下。
            void find1(char array[] , char search, char * pa)
            {
               int i;
               for (i=0;*(array+i)!=0;i++)
               {
                  if (*(array+i)==search)
                  {
                    pa=array+i
                    break;
                  }
                  else if (*(array+i)==0)
                  {
                    pa=0;
                    break;
                  }
               }
            }
            你覺得這個函數能實現所要求的功能嗎?
            調試:
            我下面調用這個函數試試。
            void main()
            {
              char str[]={“afsdfsdfdf\0”};  //待查找的字符串
              char a=’d’;   //設置要查找的字符
              char * p=0;  //如果查找到后指針p將指向字符串中查找到的第一個字符的地址。
              find1(str,a,p);  //調用函數以實現所要操作。
              if (0==p )
              {
                 printf (“沒找到!\n”);//1.如果沒找到則輸出此句
              }
              else
              {
                 printf(“找到了,p=%d”,p);  //如果找到則輸出此句
              }
            }
            分析:
            上面代碼,你認為會是輸出什么呢?
            運行試試。
            唉!怎么輸出的是:沒有找到!
            而不是:找到了,……。
            明明a值為’d’,而str字符串的第四個字符是’d’,應該找得到呀!
            再看函數定義處:void find1(char array[] , char search, char * pa)
            看調用處:find1(str,a,p);
            依我在第五篇的分析方法,函數調用時會對每一個參數進行一個隱含的賦值操作。
            整個調用如下:
               array=str;
               search=a;
               pa=p;    //請注意:以上三句是調用時隱含的動作。
               int i;
               for (i=0;*(array+i)!=0;i++)
               {
                  if (*(array+i)==search)
                  {
                    pa=array+i
                    break;
                  }
                  else if (*(array+i)==0)
                  {
                    pa=0;
                    break;
                  }
               }
            哦!參數pa與參數search的傳遞并沒有什么不同,都是值傳遞嘛(小語:地址傳遞其實就是地址值傳遞嘛)!所以對形參變量pa值(當然值是一個地址值)的修改并不會改變實參變量p值,因此p的值并沒有改變(即p的指向并沒有被改變)。
            (如果還有疑問,再看一看《函數參數的傳遞》了。)
            修正:
            void find2(char [] array, char search, char ** ppa)
            {
               int i;
               for (i=0;*(array+i)!=0;i++)
               {
                  if (*(array+i)==search)
                  {
                    *ppa=array+i
                    break;
                  }
                  else if (*(array+i)==0)
                  {
                    *ppa=0;
                    break;
                  }
               }
            }
            主函數的調用處改如下:
              find2(str,a,&p);  //調用函數以實現所要操作。
            再分析:
            這樣調用函數時的整個操作變成如下:
               array=str;
               search=a;
               ppa=&p;    //請注意:以上三句是調用時隱含的動作。
               int i;
               for (i=0;*(array+i)!=0;i++)
               {
                  if (*(array+i)==search)
                  {
                    *ppa=array+i
                    break;
                  }
                  else if (*(array+i)==0)
                  {
                    *ppa=0;
                    break;
                  }
               }
            看明白了嗎?
            ppa指向指針p的地址。
            對*ppa的修改就是對p值的修改。
            下面看一下指向指針變量的指針變量怎樣正確引用。
            用指向指針的指針變量訪問一維和二維數組。

            #include<stdio.h>
            #include<stdlib.h>
            main()
            {
            int a[3],b[2][2],*p1,*p2,**p3,i,j;

            printf("請輸入一維數組的值:\n");
            for(i=0;i<3;i++)
            scanf("%d",&a[i]);/*一維數組的輸入*/

            printf("請輸入二維數組的值:\n");
            for(i=0;i<2;i++)
            for(j=0;j<2;j++)
            scanf("%d",&b[i][j]);/*二維數組輸入*/

            printf("用指針輸出一維數組:\n");
            for(p1=a,i=0;i<3;i++)     /* 用指針輸出一維數組*/
            {
                printf("%4d",*(p1+i));
            }
            printf("\n");

            printf("用指向指針的指針變量輸出一維數組(1):\n");
            for(p1=a,p3=&p1,i=0;i<3;i++)
            printf("%4d",*(*p3+i));/*用指向指針的指針變量輸出一維數組*/
            printf("\n");
            printf("用指向指針的指針變量輸出一維數組(2):\n");
            for(p1=a;p1-a<3;p1++)/*用指向指針的指針變量輸出一維數組*/
            {
            p3=&p1;
            printf("%4d",**p3);
            }
            printf("\n");

            printf("用指針輸出二維數組:\n");
            for(i=0;i<2;i++)   /*用指針輸出二維數組*/
            {
               p2=b[i] ;
               for(int j=0;j<2;j++)
              {
                printf("%4d",*(p2+j)) ;
              }
            }
            printf("\n");

            printf("用指向指針的指針變量輸出二維數組(1):\n");
            for(i=0;i<2;i++)/*用指向指針的指針變量輸出二維數組*/
            {
            p2=b[i];
            p3=&p2;
            for(j=0;j<2;j++)
            printf("%4d",*(*p3+j));

            利用指向指針的指針變量對二維字符數組的訪問。

            #include<stdio.h>
            #include<stdlib.h>
            main()
            {
            int i;
            char * ptr;
            static char c[][16]={"clanguage","fox","computer","homepage"};
            /*二維字符數組*/
            static char *cp[]={c[0],c[1],c[2],c[3]};/*指針數組*/
            static char **cpp;/*指向字符指針的指針變量*/
            cpp=cp;/*將指針數組的首地址傳遞給指向字符指針的指針變量*/


            for(i=0;i<4;i++)/*按行輸出字符串*/
            printf("%s\n",*cpp++);
            printf("-----------\n");

            for(i=0;i<4;i++)/*按行輸出字符串*/
            {
            cpp=&cp[i];
            printf("%s\n",*cpp);
            }
            printf("-----------\n");

             
             for(i=0;i<4;i++)
             {
                ptr=c[i];
                printf("%s",ptr);
                printf("\n");
             }
             

            }

            }
            printf("\n");

             printf("用指向指針的指針變量輸出二維數組(2):\n");
            for(i=0;i<2;i++)/*用指向指針的指針變量輸出二維數組*/
            {
            p2=b[i];
            for(p2=b[i];p2-b[i]<2;p2++)
            {
            p3=&p2;
            printf("%4d",**p3);
            }
            printf("\n");
            }

            }

            posted on 2009-11-22 22:31 閱讀(356) 評論(0)  編輯 收藏 引用

            導航

            統計

            常用鏈接

            留言簿

            隨筆檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            久久激情亚洲精品无码?V| 久久一区二区三区99| 亚洲va久久久噜噜噜久久天堂 | 国内精品久久久久影院网站| 久久福利片| 奇米综合四色77777久久| 亚洲国产精品久久久久久| 欧美与黑人午夜性猛交久久久| 亚洲色欲久久久综合网东京热 | 亚洲精品乱码久久久久久自慰| 久久国产精品无码HDAV| 久久国产精品免费一区二区三区| 久久久久99这里有精品10| 7777久久亚洲中文字幕| 亚洲性久久久影院| 一本伊大人香蕉久久网手机| 国产一区二区久久久| 亚洲狠狠综合久久| 久久大香香蕉国产| 综合久久精品色| 狠狠人妻久久久久久综合| 久久久久亚洲精品无码蜜桃 | 久久精品一区二区三区中文字幕| 蜜臀久久99精品久久久久久小说| 久久精品一区二区影院 | 国产综合免费精品久久久| 久久久久99精品成人片直播| 久久不见久久见免费影院www日本| 久久精品aⅴ无码中文字字幕不卡 久久精品aⅴ无码中文字字幕重口 | 婷婷久久综合九色综合绿巨人| 久久久久人妻精品一区二区三区 | 久久不见久久见免费影院www日本| 日韩人妻无码一区二区三区久久 | 亚洲国产成人久久综合一区77| 国产精品久久成人影院| 亚洲熟妇无码另类久久久| 午夜不卡久久精品无码免费| 免费无码国产欧美久久18| 亚洲精品无码专区久久同性男| 国内精品久久久久久麻豆| 精品国产一区二区三区久久久狼|