一、数组
转载:https://www.cnblogs.com/fengliu-/p/10960387.html
1.int a[10]中a、a[0]、&a、&a[0]的关系:
- 这4个符号搞清楚了,数组相关的很多问题都有答案了。理解这些符号的时候要和左值右值结合起来,也就是搞清楚每个符号分别做左值和右值时的不同含义。
- a就是数组名。a做左值时表示整个数组的所有空间(10×4=40字节),又因为C语言规定数组操作时要独立单个操作,不能整体操作数组,所以a不能做左值;a做右值表示数组首元素(数组的第0个元素,也就是a[0])的首地址(首地址就是起始地址,就是4个字节中最开始第一个字节的地址)。a做右值等同于&a[0];
- a[0]表示数组的首元素,也就是数组的第0个元素。做左值时表示数组第0个元素对应的内存空间(连续4字节);做右值时表示数组第0个元素的值(也就是数组第0个元素对应的内存空间中存储的那个数)
- &a就是数组名a取地址,字面意思来看就应该是数组的地址。&a不能做左值(&a实质是一个常量,不是变量因此不能赋值,所以自然不能做左值。);&a做右值时表示整个数组的首地址。
- &a[0]字面意思就是数组第0个元素的首地址(搞清楚[]和&的优先级,[]的优先级要高于&,所以a先和[]结合再取地址)。做左值时表示数组首元素对应的内存空间,做右值时表示数组首元素的值(也就是数组首元素对应的内存空间中存储的那个数值)。做右值时&a[0]等同于a。
2.解释:为什么数组的地址是常量?
因为数组是编译器在内存中自动分配的。当我们每次执行程序时,运行时都会帮我们分配一块内存给这个数组,只要完成了分配,这个数组的地址就定好了,本次程序运行直到终止都无法再改了。那么我们在程序中只能通过&a来获取这个分配的地址,却不能去用赋值运算符修改它。
3.总结:
- &a和a做右值时的区别:&a是整个数组的首地址,而a是数组首元素的首地址。这两个在数字上是相等的,但是意义不相同。意义不相同会导致他们在参与运算的时候有不同的表现。
- a和&a[0]做右值时意义和数值完全相同,完全可以互相替代。
- &a是常量,不能做左值。
- a做左值代表整个数组所有空间,所以a不能做左值。
4.错误分析
- int *p; int a[5]; p = a; // 类型匹配
- int *p; int a[5]; p = &a; // 类型不匹配。p是int *, &a是整个数组的指针,也就是一个数组指针类型,不是int指针类型,所以不匹配
- &a、a、&a[0]从数值上来看是完全相等的,但是意义来看就不同了。
- 从意义上来看,a和&a[0]是数组首元素首地址,而&a是整个数组的首地址;
- 从类型来看,a和&a[0]是元素的指针,也就是int *类型;而&a是数组指针,是int (*)[5];类型。
二、字符串数组
注意指针和地址。
int main()
{
const char* pName[] = {"java","c++","python"};
setbuf(stdout, NULL);
printf("%p\n",pName);//000000000066fd90,数组首元素的地址,指向的是数组开辟的内存地址,并非字符串常量的地址,字符串常量处于静态区,暂时我还不知道以什么样的方式放于了数组开辟的内存空间里。
printf("%p\n",&pName);//000000000066fd90,数组的地址=数组首元素的地址
printf("%p\n",pName[0]);//000000000040901c,字符串常量java的地址
printf("%p\n",&pName[0]);//000000000066fd90,数组首元素的地址
// qDebug()<<pName;
// qDebug()<<reinterpret_cast<const void*>(pName[0]);
// qDebug()<<&pName[0];
const char * p = "java";
printf("%p\n",&p);//000000000066fd88,指针p存放的位置,而非指向的位置
printf("%p\n",p);//000000000040901c,字符串常量java的地址
// qDebug()<< reinterpret_cast<const void*>(p);
return 0;
}
声明一个数组时,编译器会为该数组分配内存,并为数组名分配一个固定的地址,这个地址是数组的起始地址。这个地址不能被修改,因此数组名是一个常量指针。
//是错的
char str[100];
str = "aaa";
//因为str是一个常量指针,它的值不能被改变。
//相反,你可以使用strcpy等函数来将字符串复制到数组中,这样可以实现相同的效果:
char str[100];
strcpy(str, "aaa"); // 这样是合法的