【转】C语言->数组名与指针

以下内容转自: http://bbs.csdn.net/topics/390733584

感觉对自己有用,收藏了。如果转帖,请注明如上出处。多谢!

看到一篇关于C语言指针的文章,正好最近在看《C和指针》就去看了一下。《一道 C 语言指针访存题目的引申》,里面的第一个题目就把我难住了。

帮助123456789 #include <stdio.h>   int main() {     int a[5] = {1, 2, 3, 4, 5};     int *pa = (int)(&a) + 1;     printf("%x\n", *pa);     return 0; }

正确的输出结果应该是 2000000.自诩对C指针还比较了解,不就是保存内存地址的变量吗?有什么难的,后来才发现我错了。我想的是&a,是对数组名字去地址,那出来的结果不就是保存a的地址,那这样给他转换成int变量之后再加1,然后再赋值给一个int*指针是会出错的,结果我编译了一下,真的出错了:
error: invalid conversion from `int‘ to `int*‘.
无法将int值赋值给int *指针变量,我修改了一下,在(int)(&a) + 1;前面加上了一个强制类型转换 (int *),编译之后输出结果:

我以为正确了,虽然他说正确答案是 2000000。后来我又仔细看了一下源代码,发现 int *pa = (int *)(int)(&a) + 1; 这句代码是先将 &a 转换成指针再在其基础加 1。而原来的意思是先给 &a 加 1再转换成指针,结果我再修改了一下代码:int *pa = (int *)((int)(&a) + 1); 这样结果竟然正确了,输出2000000:

这时候我完全朦了,这是怎么回事?后来我将 &a 和 a作为整数输出,想看看它到底是什么,结果发现

帮助12 printf("%d\n",&amp;a); printf("%d\n",a);

他们的输出结果是一样的,也就是说a 和 &a 的值一样,我以为问题到这里就解决了,但是我还是不能理解为什么会输出 2000000.于是我到网上去搜索了一下资料,结果搜到了 对数组名取地址是什么? 。里面说

帮助123 1 int array[100]; 3 memset(array,  0, sizeof(array)); 4 memset(&amp;array, 0, sizeof(array));

第3行和第4行有什么不同吗?其实从效果上来说是一样的,但是这里要注意 array 和 &array 的类型是不同的。array 相当于 &array[0],而 &array 是一个指向 int[100] 的指针,类型是 int(*)[100]。”

总算知道了a和&a的差别,但是我还是不明白为什么它会输出 2000000,这到底是为什么呢?

我试着修改了一下代码:

int a[] = {1,5};

这样之后的输出结果成了 : 5000000 原来和数组a的第二个元素有关系。哈哈,有点懂了,经过我N次的试验。

首先我们的代码是:

帮助123 int a[] = {1,2};     int *pa = (int *)((int)(&amp;a) + 1);     printf("%x\n", *(pa));

我这边编译的结果 a[0]也就是a代表的地址是 0x22ff70 a[1] 的地址是 0x22ff74 。我查看了一下内存(使用printf看的),内存里的数据是这样的:

帮助12 0x22ff70 0x22ff71 ... 0x22ff74 ... 0x22ff77 01          00   ...      02   ...    00

而且我们知道访问内存中的数据是高字节的地址是高位,低字节的地址是低位。也就是说如果我的地址是 0x22ff70,以这个地址访问一个整数,得到的整数就是73,72,71,70的数据排列 即00000001(以前学过一点汇编,加上自己试验了一下,就出来的 :),也就是1。如果访问0x22ff74 也就是 00000002 是2.如果访问 0x22ff71 也就是从74开始 02000000。这不就是我们的正确答案 2000000 。
问题解决!写得比较乱,勿怪。若有错误,欢迎指正!

时间: 2024-11-02 10:48:18

【转】C语言->数组名与指针的相关文章

对于C语言中数组名是指针的理解

我们都知道,c语言中数组名是一个指针,比如下面这段代码 #include<iostream>using namespace std;int main(){ int a[4]={1,2,3,4}; for(int i=0;i<4;i++) {  cout<<*(a+i);//*(a+i)和a[i]是等价的.  cout<<endl; } return 0;} 但是看下面这个代码 #include<iostream>using namespace std;

C 语言中的左值和右值。以及对比数组名和指针取数组元素的区别。

左值:出现在赋值符左边的符号有时称为左值. 右值:出现在赋值符右边的符号有时称为右值. 编译器为每个变量分配一个地址(左值),这个地址在编译时可知,而且该变量在运行时一直保存于这个地址.相反,存储于变量中的值(它的右值)只有在运行时才可知.如果需要用到变量中存储的值,编译器就发出指令从指定地址读入变量值并将它存于寄存器. 可以看到,每个符号的地址在编译时可知. 对比一下几个式子: //常规变量 int a=1;//这里a作为左值出现,代表的是地址,即在a表示的这个内存地址存入数值1.即a代表的内

C语言,数组名,指针常量和常量指针

自己给自己劝退一波~ 先介绍指针常量和常量指针 const int *p; //这是个指针常量 int const* p; //这是个常量指针 const是常量修饰符,被修饰的就是一个常量,常量的特性就是不能改变. 先介绍指针常量, const int *p; int i=2; p=&i; 这里*p就是一个常量,他的值不能改变,也就是不能再通过*p=1; 这样赋值而去改变 i 的值,但可以通过直接i=1;这样改变i的值,此时*p的值也相应变为了1 int const* p=NULL; int i

c/c++数组名和指针区别深入探索

指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很多程序设计者就被搞糊涂了.而许多的大学老师,他们在C语言的教学过程中也错误得给学生讲解:"数组名就是指针".很幸运,我的大学老师就是其中之一.时至今日,我日复一日地进行着C/C++项目的开发,而身边还一直充满这样的程序员,他们保留着"数组名就是指针"的误解. 想必这种误解的根源在于国内某著名的C程序设计教程.如果这篇文章能够纠正许多中国程序员对数组名和指针的误解,

数组名和指针的深入理解(C++)

指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很多程序设计者就被搞糊涂了. 魔幻数组名 请看程序(本文程序在WIN32平台下编译): #include <iostream> using namespace std; int main() { char str[10]; char* pStr = str; cout << "sizeof(str): \t" << sizeof(str) <&

第六章 数组名与指针

指针的算术运算符是指针和数组之间的一种关联,但不是唯一关联: 可以使用数组名作为指向数组第一个元素的指针,但是不可以给数组名赋新的值. //如下声明a int a[10]; //用a作为指向数组第一个元素的指针,可以修改a[0]; *a = 11; //通过a + 1来访问a[1]; *(a + 1) = 22; 但是试图使数组名指向其他地方是错误的: #include <stdio.h> int main() { char arr[] = "yangxunwu"; whi

[skill] C语言数组名到底是个啥

1. 正常情况下,数组名是个地址常量. 2. sizeof(数组名)的时候,数组名就代表数字名,其类型为 type array[], 返回数组元素个数. 3. 除了2的情况以外,可以理解为一个指针常量 !? 参考: http://www.cnblogs.com/kalo1111/p/3285955.html http://c.biancheng.net/cpp/html/492.html

二维数组中数组名与其指针的关系

大多数初学者不是很清楚二维数组是个怎样的存在,今天我就来说说这个二维数组,计算机内存是一连串的存储单元,我们可以把它理解成一条线,没错就是一条线,那么问题来了,二维是个怎样的存在呢???说白了,二维数组并不真是二维,只是人们主观的将它变成二维,举个例子吧!比如楼梯,我们都知道楼梯说白了就是一条路,但是我们刻意将楼梯来回上升的方式,这都是我们主观的改变,实际上我们可以看成是一个直路,只是被工人弄弯了.好了,不扯了,进入主题.... 二维数组名a是数组第一行的指针,它指向的是一整行,*a才是a[0]

二维数组名和指针

#include <iostream> using namespace std; #define M 2 #define N 3 int main() { int a[M][N] = {1,2,3,4,5,4}; cout<<&(a[0])<<endl; //00DCFA64 cout<<&(a[0])+1<<endl; //00DCFA70,offest:12 cout<<a<<endl; //00DCF