上周五参加Realtek的笔试题,有一道选择题,大概是这样的:
const char a1[] = "abc"; const char a2[] = "abc"; const char *p1 = "abc"; const char *p2 = "abc";
A、a1和a2一样,p1和p2不一样;
B、a1和a2不一样,p1和p2一样;
C、a1和a2不一样,p1和p2也不一样;
D、a1和a2一样,p1和p2一样;
我在VS2005的环境下,将代码敲进然后仿真,发现:
a1=0x004156b0;
a2=0x004156b4;
p1=0x0041563c;
p2=0x0041563c;
由此可见,正确答案应该是B。
那么这题为什么要选B呢?后来问了一下,得到的结论是这样的:
例如:const char a1[] = "abc";是这样的一个步骤:
//当然实际代码中这样拆分开来写,应该是不对的,这里只是为了说明这个过程而已
1、const char a1[];//这个时候系统分配了一个空间,该空间的首地址即数组名a1
2、a[1] = "abc";//将字符串装进分配的内存单元之中
由此看来前两种都是先分配空间,至于空间里面装什么跟分配空间的首地址无关,也就是说编译器会分配两个首地址不同的空间给两个数组,这样两个数组的首地址当然就不同了,所以a1和a2是不同的。
而后两种,例如:const char *p1 = "abc";却是这样的一个过程:
//当然实际代码中这样拆分开来写,应该是不对的,这里只是为了说明这个过程而已
1、const char *p1;
2、p1 = “abc”;
也就是说,我们先定义一个char*的指针,然后将“abc”赋给该指针。那么这里“abc”到底是什么呢?很显然是一个常量字符串,而该常量字符串的值实际上是一个地址,指向该字符串。也就是说“abc”实际上是一个地址,而且由于该字符串是一个常量字符串,所以该指针是个常量。
那么无论p1= “abc”,p2 = “abc” 还是p3 = “abc”,都是将这个指针常量赋给了p1,p2和p3,所以p1=p2;
今天在网上搜索相关问题时,看到有人提问如下:
int isequiv(char *t) { if (t == "char") return 1; else return 0; } int main() { char test[] = "char"; char *temp = "char"; printf("%d", isequiv(test));//输出 0 printf("%d", isequiv(temp));//输出 1 getchar(); return 0; }
提问的同学想问为什么两个printf输出的不一样,在他看来应该是一样的。根据我在上面的解释,这题输出很明显是不一样的。
最后还在网上看到了一个小的知识点,是一个关于字符串常量指针的问题,原文如下:
采用指针存储字符串,其实质就是把字符串的首地值附给基类型为char的指针变量,从而可以从字符串首元素开始对字符串进行操作,这里面也存在一点问题.
用这个类子给大家解释解释.
int main()
{
char *p="hello world";
p[0]=‘H‘;
printf("%s\n",p);
return 0;
}
运行结果会出现断错误,原因在于,*p="hello world" 这句仅仅声明了一个指针变量,指向字符串"hello world",而"hello world"这个字符串程序没有给它分配空间,编译器把它分配到常量区.而常量字符串的值是不允许被修改的 ,所以会出现断错误.
程序改为如下就正确了
int main()
{
char p[12]="hello world";
char *p1=p;
p1[0]=‘H‘;
printf("%s\n",p1);
return 0;
}
原因在于,p[12]="hello world"是你自己定义的一个长度为12 的字符数组,所以字符串"hello world"编译器会给它分配空间(在栈中),所以你能修改它的值.
最后我在《C和指针》p153页末看到这样的内容:
char message1[] = "Hello";
char *message2 = "Hello";
这两个初始化看上去很像,但他们具有不同的含义。前者是初始化一个字符数组的元素,而后者则是一个真正的字符串常量。这个指针变量被初始化为指向这个字符串常量的存储位置。