程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> 關於C++ >> C/C++中字符指針數組及指向指針的指針的含義

C/C++中字符指針數組及指向指針的指針的含義

編輯:關於C++

就指向指針的指針,很早以前在說指針的時候說過,但後來發現很多人還是比較難以理解,這一次我們再次仔細說一說指向指針的指針。

先看下面的代碼,注意看代碼中的注解:

#include <iostream>
#include <string>
using namespace std;

void print_char(char* array[],int len);//函數原形聲明
void main(void)
{
//-----------------------------段1-----------------------------------------
char *a[]={"abc","cde","fgh"};//字符指針數組
char* *b=a;//定義一個指向指針的指針,並賦予指針數組首地址所指向的第一個字符串的地址也就是abc\0字符串的首地址
cout<<*b<<"|"<<*(b+1)<<"|"<<*(b+2)<<endl;
//-------------------------------------------------------------------------
//-----------------------------段2-----------------------------------------
char* test[]={"abc","cde","fgh"};//注意這裡是引號,表示是字符串,以後的地址每加1就是加4位(在32位系統上)
int num=sizeof(test)/sizeof(char*);//計算字符串個數
print_char(test,num);
cin.get();
//-------------------------------------------------------------------------
}

void print_char(char* array[],int len)//當調用的時候傳遞進來的不是數組,而是字符指針他每加1也就是加上sizeof(char*)的長度
{
for(int i=0;i<len;i++)
{
cout<<*array++<<endl;
}
}

下面我們來仔細說明一下字符指針數組和指向指針的指針,段1中的程序是下面的樣子:

char *a[]={"abc","cde","fgh"};
char* *b=a;
cout<<*b<<"|"<<*(b+1)<<"|"<<*(b+2)<<endl;

char *a[]定義了一個指針數組,注意不是char[], char[]是不能同時初始化為三個字符的,定義以後的a[]其實內部有三個內存位置,分別存儲了abc,cde,fgh,三個字符串的起始地址,而這三個位置的內存地址卻不是這三個字符串的起始地址,在這個例子中a[]是存儲在棧空間內的,而三個字符串卻是存儲在靜態內存空間內的const區域中的,接下去我們看到了char* *b=a;這裡是定義了一個指向指針的指針,如果你寫成char *b=a;那麼是錯誤的,因為編譯器會返回一個無法將char* *[3]轉換給char *的錯誤,b=a的賦值,實際上是把a的首地址賦給了b,由於b是一個指向指針的指針,程序的輸出cout<<*b<<"|"<<*(b+1)<<"|"<<*(b+2)<<endl;

結果是

abc
cde
fgh

可以看出每一次內存地址的+1操作事實上是一次加sizeof(char*)的操作,我們在32位的系統中sizeof(char*)的長度是4,所以每加1也就是+4,實際上是*a[]內部三個位置的+1,所以*(b+1)的結果自然就是cde了,我們這時候可能會問,為什麼輸出是cde而不是c一個呢?答案是這樣的,在c++中,輸出字符指針就是輸出字符串,程序會自動在遇到後停止.

我們最後分析一下段2中的代碼,段2中我們調用了print_array()這個函數,這個函數中形式參數是char *array[]和代碼中的char *test[]一樣,同為字符指針,當你把參數傳遞過來的時候,事實上不是把數組內容傳遞過來,test的首地址傳遞了進來,由於array是指針,所以在內存中它在棧區,具有變量一樣的性質,可以為左值,所以我們輸出寫成了,cout<<*array++<<endl;當然我們也可以改寫為cout<<array[i]<<endl,這裡在循環中的每次加1操作和段1代碼總的道理是一樣的,注意看下面的圖!

到這裡這兩個非常重要的知識點我們都說完了,說歸說,要想透徹理解希望讀者多動手,多觀察,熟能生巧。

下面是內存結構示意圖:

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved