今天写个小文说一说字符串地址和字符串常量。
在C/C++中,一个字符串常量表示的是该字符串第一个元素的地址,就跟char指针名,char数组名表示的是字符串第一个元素的地址一样。
想要打印一个地址,用一个简单的 cout << 地址; 语句就可以搞定;
但是下面这两条语句将打印整个字符串
char a[20] = "1234"; cout << a << endl; char *p = a; cout << p << endl;
这也是字符数组 与其他数组不同的一个地方,那么该如何得到该字符串的地址呢?
下面有两种方法可供参考
cout << (int*)a << endl; cout << &a << endl;
这两种方法都可以正确打印出“字符串的地址”,但是有细微区别之处
在字符数组a中,a表示第一个字符的地址,a+1表示第二个字符的地址;
在第一条打印地址的语句中,(int*)a只是起到了一个强制类型转换的作用,换句话说,a表示第一个字符的地址,但是cout <<a;输出的是整个字符串,这是因为这个地址是char*类型的,cout识别到char*类型的地址将会自动打印从该地址指向的空间开始直到遇到'\0'的字符串内容,所以这里我们只需要进行一个强制类型转换(这里举例强制转换为int*,可以转换为其他类型的指针,只要不是char*,哪怕是double*也可以正确打印地址)。所以如果想要第二个字符的地址呢,cout << (int*)(a+1);就可以啦
第二条打印地址的语句中,直接使用了&a;这里需要强调的是,不论任何数组,数组名表示的都是第一个元素的地址,&数组名表示的才是该数组的地址,虽然二者打印出来是同一地址,但是如果进行指针运算则天差地别,看一下下面这段代码和效果
int a[10]; cout << "a is: " << a << " a+1 is: " << a+1 << endl; cout << "&a is: " << &a << " &a+1 is: " << &a+1 << endl;
有图有真相,a+1只是向后移动了4个字节,而&a+1直接向后移动了40个字节,所以&数组名表示的才是整个数组的地址。所以同理用&字符数组名也可以打印出字符串地址。
下面说一说字符串常量,首先字符串存储在静态存储区,其次,不可修改。
于是就产生了下面几条语句
char a[20] = "1234"; char * p = "1234"; const char * p = "1234";
第一条语句不用说,很常用了,正确,那么二三条语句哪个对哪个错呢。
----------------------------以下为9.7号加更----------------------------
很明显,像"1234"这样的字符串常量被存储在静态存储区中,他们的类型是const char*型,根据类型匹配原则,很显然将其赋给一个char * 的指针肯定是不行的。第三条语句百分百正确。
但是第二条语句在c 和 c++两种环境下编译产生了微小差异。
在C环境下编译程序正常运行,且0warning.
而在C++下,则产生了一条警告
这是因为在传统的原始C中,是没有const的概念的,也就是说在C中没有硬性规定常量"1234"不能修改,但是再进一步尝试着修改时,会产生内存错误,也就是说字符串常量"1234"是const char*类型的,不能被修改
而在C++中规范了这种情况,明确规定"常量1234"就存储于静态存储区,这里没有报错而只是抛出一条警告仅仅是为了兼容C。
所以不论是在C还是C++中建议使用const char*指向一个字符串常量。
------------------------------end-------------------------------------------
刚开始我还觉得加不加const无所谓,反正都是错的,后来细思极恐,果然还是知识贫穷限制了想象力;
一开始我认为后面两句错的原因是因为 没见过,看着像是给*p赋了个"1234",可又不是,这是个什么东西,总之就是看着怪怪的;
所以这里就牵扯到开头提到的一个知识点了;在C/C++中,一个字符串常量表示的是该字符串第一个元素的地址,就跟char指针名,char数组名表示的是字符串第一个元素的地址一样
所以这回就明白了,看着是"1234",其实本质上是"1234"第一个字符的地址,那么将 char*型的指针p指向“1234”就没有问题啦,地址对地址嘛,但是编译后出错了,理由是"1234"是一个常量,所以加上了const限定。
关于这里为什么要加const限定才正确,如果不加的话类型也是匹配的,为什么会报错呢,所以这里需要再强调一下,"1234"是字符串常量,常量不允许被修改,如果char * p = "1234";编译成功,那就意味着"1234"有可能被修改,而这是不被允许的,所以必须乖乖听话加上const限定符让系统安心才是嘛。
咦?常量不允许被修改,要加const限定符才可以,那为什么我们一直用的第一条语句是正确的而且从来没有报错过呢?
这就要说说第一条语句和第三条语句的区别啦,"1234"是字符串常量,常量存储在静态存储区,有地址
第三条语句是让一个常量指针直接指向静态存储区的"1234"的地址,因为是直接指向本尊地址,所以要加上const限定符
而第一条语句则是将静态存储区的"1234"的副本拿出来复制到字符数组a中,所以这样初始化可以在后面随意更改副本"1234",这样并不会影响到真正的"1234";
是真地址假地址cout一下就知道了
1 char a[20] = "1234"; 2 const char * p = "1234"; 3 cout << &a << endl; // 获取字符数组a的地址 4 cout << &"1234" << endl; // 获取 "1234"在静态存储区的地址 5 6 //再来验证一下字符串常量为该字符串第一个元素的地址 7 cout << *"1234" << endl; 8 cout << *("1234"+1) << endl;
OK,明显副本”1234“与本尊”1234“的地址不同,而且差了很多很多地址位,肯定是跨内存区了的,最后面两行想要的结论也得到了验证,实践出真知,真是不枉我熬夜到12点。