C語言中對于下面的兩種情況,是否相同呢?
char a[] = "abcdefg";---------------1
char *p = "abcdefg";-----------------2
在談到這些區(qū)別時(shí),應(yīng)該先談一下計(jì)算機(jī)中對變量是如何存儲(chǔ)的。從編譯原理中我們知道,對于所有的變量他都會(huì)影射到一個(gè)符號(hào)表中。為了簡化,這里給出一種最簡單的便于理解的符號(hào)表:
表1 一個(gè)簡單的符號(hào)表示例
以上表格中a代表一個(gè)變量,0xffaa則為變量a的內(nèi)容的存儲(chǔ)地址;p代表另一個(gè)變量,0xffcc為變量p的內(nèi)容的存儲(chǔ)地址。對于數(shù)組型的變量和指針型的變量,其地址代表的含義不同。
對于數(shù)組a:
這個(gè)0xffaa地址就是其存放數(shù)組內(nèi)容的首地址了。對于a[i]的引用步驟如下:
步驟一、取出i的值,將他與0xffaa相加;
步驟二、取出為(0xffaa+i)中的內(nèi)容。
對于指針p:
這個(gè)0xffcc地址就是中存放的不是字符串的內(nèi)容,而是一個(gè)地址,這個(gè)地址才是字符串的首地址,對p[i]或者用指針表示*(p+i)的應(yīng)用步驟如下:
步驟一、取出0xffcc地址中的內(nèi)容,例如為0xffdf;
步驟二、取出地址0xffdf中的內(nèi)容。
數(shù)組和指針的對比如下圖:
下面是在VC6.0下作的一個(gè)試驗(yàn),通過這個(gè)試驗(yàn)大家可以看到,雖然同過[]和通過*引用都一樣,但在內(nèi)部處理的方法是不一樣的。
#include "stdafx.h"
#include "stdio.h"
int main(int argc, char* argv[])
{
int a[3]={1,2,3};
int *p =a;
printf("a:%d,&a:%d,a[0]:%d,*a:%d,p:%d,&p:%d,*p:%d,p[0]:%d",a,&a,
a[0],*a,p,&p,*p,p[0]);
return 0;
}
輸出結(jié)果:
a:1310580,&a:1310580,a[0]:1,*a:1,p:1310580,&p:1310576,*p:1,p[0]:1。
由上面的分析可知,如果在一個(gè)文件中定義了一個(gè)數(shù)組int maychar[100],那么下面的聲明就是完全錯(cuò)誤的。
extern int *maychar;
這樣的話,在引用時(shí)他就會(huì)按照指針的方法來引用數(shù)組。正確的聲明應(yīng)該是exter int maychar[];這里數(shù)組的大小并不重要。下面將指針與數(shù)組的區(qū)別用表格的形式列出如下:
指針 | 數(shù)組 |
保存數(shù)據(jù)的地址 | 保存數(shù)據(jù) |
間接訪問數(shù)據(jù) | 直接訪問 |
通常用于動(dòng)態(tài)數(shù)據(jù)結(jié)構(gòu) | 通常用于存儲(chǔ)固定數(shù)目數(shù)據(jù)類型相同的元素 |
相關(guān)操作malloc(),free()等 | 隱式分配和刪除 |
同常指向匿名數(shù)據(jù) | 自身即為數(shù)據(jù)名 |
表2 指針與數(shù)組的區(qū)別
還要提醒一點(diǎn)的就是:
char a[] = "abcdefg";---------------數(shù)組內(nèi)容能修改(字符數(shù)組)
char *p = "abcdefg";-----------------內(nèi)容不能修改(字符串常量)
在ANSI C中,初始化指針是所創(chuàng)建的字符串時(shí)常量,被定義為只讀,如果試圖通過指針修改這個(gè)字符串的值,程序就會(huì)出現(xiàn)為定義的行為。