雷火电竞-中国电竞赛事及体育赛事平台

歡迎來到入門教程網(wǎng)!

C語言

當前位置:主頁 > 軟件編程 > C語言 >

圖文詳解c/c++中的多級指針與多維數(shù)組

來源:本站原創(chuàng)|時間:2020-01-10|欄目:C語言|點擊:

前言

首先先聲明一些常識,如果你對這些常識還不理解,請先去彌補一下基礎(chǔ)知識:

     1、實際上并不存在多維數(shù)組,所謂的多維數(shù)組本質(zhì)上是用一維數(shù)組模擬的。

     2、數(shù)組名是一個常量(意味著不允許對其進行賦值操作),其代表數(shù)組首元素的首地址。

     3、數(shù)組與指針的關(guān)系是因為數(shù)組下標操作符[],比如,int a[3][2]相當于*(*(a+3)+2) 。

     4、指針是一種變量,也具有類型,其占用內(nèi)存空間大小和系統(tǒng)有關(guān),一般32位系統(tǒng)下,sizeof(指針變量)=4。

     5、指針可以進行加減算術(shù)運算,加減的基本單位是sizeof(指針所指向的數(shù)據(jù)類型)。

     6、對數(shù)組的數(shù)組名進行取地址(&)操作,其類型為整個數(shù)組類型。

     7、對數(shù)組的數(shù)組名進行sizeof運算符操作,其值為整個數(shù)組的大小(以字節(jié)為單位)。

     8、數(shù)組作為函數(shù)形參時會退化為指針。

 一、一維數(shù)組與數(shù)組指針

      假如有一維數(shù)組如下:

 char a[3];

      該數(shù)組一共有3個元素,元素的類型為char,如果想定義一個指針指向該數(shù)組,也就是如果想把數(shù)組名a賦值給一個指針變量,那么該指針變量的類型應該是什么呢?前文說過,一個數(shù)組的數(shù)組名代表其首元素的首地址,也就是相當于&a[0],而a[0]的類型為char,因此&a[0]類型為char *,因此,可以定義如下的指針變量: 

 char * p = a;//相當于char * p = &a[0]

      以上文字可用如下內(nèi)存模型圖表示。

 

      大家都應該知道,a和&a[0]代表的都是數(shù)組首元素的首地址,而如果你將&a的值打印出來,會發(fā)現(xiàn)該值也等于數(shù)組首元素的首地址。請注意我這里的措辭,也就是說,&a雖然在數(shù)值上也等于數(shù)組首元素首地址的值,但是其類型并不是數(shù)組首元素首地址類型,也就是char *p = &a是錯誤的。

      前文第6條常識已經(jīng)說過,對數(shù)組名進行取地址操作,其類型為整個數(shù)組,因此,&a的類型是char (*)[3],所以正確的賦值方式如下:

 char (*p)[3] = &a;

      注:很多人對類似于a+1,&a+1,&a[0]+1,sizeof(a),sizeof(&a)等感到迷惑,其實只要搞清楚指針的類型就可以迎刃而解。比如在面對a+1和&a+1的區(qū)別時,由于a表示數(shù)組首元素首地址,其類型為char *,因此a+1相當于數(shù)組首地址值+sizeof(char);而&a的類型為char (*)[3],代表整個數(shù)組,因此&a+1相當于數(shù)組首地址值+sizeof(a)。(sizeof(a)代表整個數(shù)組大小,前文第7條說明,但是無論數(shù)組大小如何,sizeof(&a)永遠等于一個指針變量占用空間的大小,具體與系統(tǒng)平臺有關(guān))

二、二維數(shù)組與數(shù)組指針

      假如有如下二維數(shù)組:

 char a[3][2];

      由于實際上并不存在多維數(shù)組,因此,可以將a[3][2]看成是一個具有3個元素的一維數(shù)組,只是這三個元素分別又是一個一維數(shù)組。實際上,在內(nèi)存中,該數(shù)組的確是按照一維數(shù)組的形式存儲的,存儲順序為(低地址在前):a[0][0]、a[0][1]、a[1][0]、a[1][1]、a[2][0]、a[2][1]。(此種方式也不是絕對,也有按列優(yōu)先存儲的模式)

      為了方便理解,我畫了一張邏輯上的內(nèi)存圖,之所以說是邏輯上的,是因為該圖只是便于理解,并不是數(shù)組在內(nèi)存中實際的存儲模型(實際模型為前文所述)。

    

      如上圖所示,我們可以將數(shù)組分成兩個維度來看,首先是第一維,將a[3][2]看成一個具有三個元素的一維數(shù)組,元素分別為:a[0]、a[1]、a[2],其中,a[0]、a[1]、a[2]又分別是一個具有兩個元素的一維數(shù)組(元素類型為char)。從第二個維度看,此處可以將a[0]、a[1]、a[2]看成自己代表”第二維”數(shù)組的數(shù)組名,以a[0]為例,a[0](數(shù)組名)代表的一維數(shù)組是一個具有兩個char類型元素的數(shù)組,而a[0]是這個數(shù)組的數(shù)組名(代表數(shù)組首元素首地址),因此a[0]類型為char *,同理a[1]和a[2]類型都是char *。而a是第一維數(shù)組的數(shù)組名,代表首元素首地址,而首元素是一個具有兩個char類型元素的一維數(shù)組,因此a就是一個指向具有兩個char類型元素數(shù)組的數(shù)組指針,也就是char(*)[2]。

     也就是說,如下的賦值是正確的:

 char (*p)[2] = a;//a為第一維數(shù)組的數(shù)組名,類型為char (*)[2]

 char * p = a[0];//a[0]維第二維數(shù)組的數(shù)組名,類型為char *

      同樣,對a取地址操作代表整個數(shù)組的首地址,類型為數(shù)組類型(請允許我暫且這么稱呼),也就是char (*)[3][2],所以如下賦值是正確的: 

 char (*p)[3][2] = &a;

三、三維數(shù)組與數(shù)組指針

     假設(shè)有三維數(shù)組:

 char a[3][2][2];

     同樣,為了便于理解,特意畫了如下的邏輯內(nèi)存圖。分析方法和二維數(shù)組類似,首先,從第一維角度看過去,a[3][2][2]是一個具有三個元素a[0]、a[1]、a[2]的一維數(shù)組,只是這三個元素分別又是一個"二維"數(shù)組,a作為第一維數(shù)組的數(shù)組名,代表數(shù)組首元素的首地址,也就是一個指向一個二維數(shù)組的數(shù)組指針,其類型為char (*)[2][2]。從第二維角度看過去,a[0]、a[1]、a[2]分別是第二維數(shù)組的數(shù)組名,代表第二維數(shù)組的首元素的首地址,也就是一個指向一維數(shù)組的數(shù)組指針,類型為char(*)[2];同理,從第三維角度看過去,a[0][0]、a[0][1]、a[1][0]、a[1][1]、a[2][0]、a[2][1]又分別是第三維數(shù)組的數(shù)組名,代表第三維數(shù)組的首元素的首地址,也就是一個指向char類型的指針,類型為char *。

            由上可知,以下的賦值是正確的:

  char (*p)[3][2][2] = &a;//對數(shù)組名取地址類型為整個數(shù)組
  char (*p)[2][2] = a;
  char (*p) [2] = a[0];//或者a[1]、a[2]
  char *p = a[0][0];//或者a[0][1]、a[1][0]...

四:多級指針

      所謂的多級指針,就是一個指向指針的指針,比如:

  char *p = "my name is chenyang.";

  char **pp = &p;//二級指針

  char ***ppp = &pp;//三級指針

      假設(shè)以上語句都位于函數(shù)體內(nèi),則可以使用下面的簡化圖來表達多級指針之間的指向關(guān)系。

     

         多級指針通常用來作為函數(shù)的形參,比如常見的main函數(shù)聲明如下:

 int main(int argc,char ** argv)

         因為當數(shù)組用作函數(shù)的形參的時候,會退化為指針來處理,所以上面的形式和下面是一樣的。

 int mian(int argc,char* argv[]) 

         argv用于接收用戶輸入的命令參數(shù),這些參數(shù)會以字符串數(shù)組的形式傳入,類似于:

 char * parm[] = {"parm1","parm2","parm3","parm4"};//模擬用戶傳入的參數(shù)

 main(sizeof(parm)/sizeof(char *),parm);//模擬調(diào)用main函數(shù),實際中main函數(shù)是由入口函數(shù)調(diào)用的(glibc中的入口函數(shù)默認為_start)

         多級指針的另一種常見用法是,假設(shè)用戶想調(diào)用一個函數(shù)分配一段內(nèi)存,那么分配的內(nèi)存地址可以有兩種方式拿到:第一種是通過函數(shù)的返回值,該種方式的函數(shù)聲明如下:

 void * get_memery(int size)
 {
  void *p = malloc(size);
  return p;
  }

        第二種獲取地址的方法是使用二級指針,代碼如下:

 int get_memery(int** buf,int size)
 { 
  *buf = (int *)malloc(size);
  if(*buf == NULL)
   return -1;
  else
   return 0;
 }
  int *p = NULL;
  get_memery(&p,10);

總結(jié)

關(guān)于多級指針的用法很多,尤其以二級指針應用最為廣泛,后續(xù)的有時間再進行補充。c/c++中的多級指針與多維數(shù)組的內(nèi)容到這就基本結(jié)束了,希望本文的內(nèi)容對大家能有所幫助。

上一篇:C語言 常量詳解及示例代碼

欄    目:C語言

下一篇:如何應用C++的函數(shù)對象

本文標題:圖文詳解c/c++中的多級指針與多維數(shù)組

本文地址:http://www.jygsgssxh.com/a1/Cyuyan/2107.html

網(wǎng)頁制作CMS教程網(wǎng)絡編程軟件編程腳本語言數(shù)據(jù)庫服務器

如果侵犯了您的權(quán)利,請與我們聯(lián)系,我們將在24小時內(nèi)進行處理、任何非本站因素導致的法律后果,本站均不負任何責任。

聯(lián)系QQ:835971066 | 郵箱:835971066#qq.com(#換成@)

Copyright © 2002-2020 腳本教程網(wǎng) 版權(quán)所有