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代码总的原理是一样的,留意看下面的图!
到这里这两个很是重要的常识点我们都说完了,说归说,要想透彻领略但愿读者多动手,多调查,熟能生巧。
下面是内存布局示意图: