C/C++中字符指针和字符数组、字符指针数组及指向指针的指针的含义

来源:互联网 发布:机械键盘测试软件 编辑:程序博客网 时间:2024/06/04 19:27

字符指针和字符数组

char* s1="hello";

char  s2[]="world";
两个表达式的含义是一样的.
如果讲不同之处的话,VC6中编译时,对这两种定义进行了不同的处理;
char *s1 = "hello"; 中的"hello",编译时指针指向的区域位于PE文件的.rdata节中,是只读的.
不信的话,你可以试试:

main()
{
  char *s1="hello";
  char s2[]="world";

  *(s2+2)='x'; //正确
  *(s1+2)='x'; //执行时出错.
}

1.严格的说两个表达的意思是不完全一样的,因为前者是个字符串指针,这个指针S1所存的地址就是存储字符串前8个字节即hello/n/n/n的那个地址。
  2.而后者是字符数组。每个字符都有一个独立的地址。
见图示。
 

字符指针数组及指向指针的指针的含义

#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/0,cde/0,fgh/0,三个字符串的起始地址,而这三个位置的内存地址却不是这三个字符串的起始地址,在这个例子中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++中,输出字符指针就是输出字符串,程序会自动在遇到/0后停止.

  我们最后分析一下段2中的代码,段2中我们调用了print_array()这个函数,这个函数中形式参数是char *array[]和代码中的char *test[]一样,同为字符指针,当你把参数传递过来的时候,事实上不是把数组内容传递过来,test的首地址传递了进来,由于array是指针,所以在内存中它在栈区,具有变量一样的性质,可以为左值,所以我们输出写成了,cout<<*array++<<endl;当然我们也可以改写为cout<<array[i]<<endl,这里在循环中的每次加1操作和段1代码总的道理是一样!