回忆青牛的笔试题

来源:互联网 发布:打立出局淘汰赛算法 编辑:程序博客网 时间:2024/05/17 06:02
【zz】1 第一个 void main() { int a[5] = {1,2,3,4,5}; int *ptr = (int *)(&a + 1); int *ptr2 = a+1; printf("%d,%d,%d/n", *(a+1), *(ptr-1), *(ptr2-1)); } 输出为2,5,1 问题:中间那个5是怎么来的?int a[5] = {1,2,3,4,5}; 此语句会让系统在栈内存中分配5个连续的int型(32位机是四个字节)空间 a是一个指针常量,a应该是被分配在静态存储区(不敢肯定,望大牛解惑 当然不是,是存在栈空间的动态存储的,这我自己知道),并且有一个固定的值假设该数组的首地址为4000,即a的值是4000,那麽数组元素5的地址应该是&a[4] = 4000+4*4 = 4016 另外假设a的地址是3000 int *ptr = (int *)(&a + 1); 指针做加法时,我们要关心的不是指针本身,而是指针所指向的数据类型很明显&a是一个指针(指针和地址的概念是等价的,若对这句话有异议,我们可以一起讨论),这个指针所指向的是a 。而a是一个数组类型(注意,这里不能将a理解为一个指针常量,数组和指针的区别还是很大的),sizeof(a) = 20 所以 ptr = &a+1 = 3000+20 = 3020,在将其转换为int*型 所以ptr-1 = 3020 - 4 = 3016, 这片内存中的数是不确定的, 所以第二个打印出来5完全是碰巧的,要不就是编译器的问题 记住,数组就是数组,指针就是指针,千万不要混用(虽然有时候混用是正确的). 只有当数组以实参传递给函数时,它才会变成指针。 这道题我们应该这样理解: a是一个int[5]型的数组,那么&a自然是指向这个数组(千万不要以为&a指向一个指针常量,这就是上面所犯的错误),那么&a+1自然是指向下一个数组 所以 ptr = 4000 + 20 = 4020 , 而不是3016。 第二题 void main() { char a[][3] = {1,2,3,4,5}; cout<
原创粉丝点击