Chinaunix首页 | 论坛 | 博客
  • 博客访问: 345293
  • 博文数量: 82
  • 博客积分: 2602
  • 博客等级: 少校
  • 技术积分: 660
  • 用 户 组: 普通用户
  • 注册时间: 2007-08-10 08:48
文章分类

全部博文(82)

文章存档

2008年(17)

2007年(65)

分类: C/C++

2007-08-31 08:54:34

自己写了一些总结,关于指针与数组方面的内容.

 

首先看指针:

char *p = “helloworld”;

在这里p是一个变量,其类型为指针类型,并且指向一个字符串,字符串内容为”helloworld”,如果要访问p[2]的话,就需要先从p中取出地址,该地址为”helloworld”的首地址,然后再加上偏移量2,就得到了’l’这个字符,所以其访问的方式为先从p中取出地址,然后再将该地址加上偏移量,得到一个地址,最后从这个地址中取出值来.

其分为三步:

p的值,该值即为字符串的首地址.

该地址加上偏移量,得到所要取的字符的地址.

从这个地址中取得值.

此处p是一个变量,它自己是存放在一个地址中的,而它的内容则是”helloworld”这个字符串的地址. p与字符串是分开的.

同时,该指针的值是动态确定的,必须在运行的时候才能确定其值,并通过该值访问到字符串.

 

而如果是数组的话,则为

char p[20] = “helloworld”;

在这里p为一个字符串的标识,其类型是一个字符数组的类型,且该数组有20char类型的大小.此时如果要访问p[2]的话,就直接使用该字符数组的首地址加上2char类型的大小的长度就得到了要访问的字符的地址,最后再从这个地址中取出值来.而且此时p的地址为该字符数组的首地址,其内容为’h’.一个字符类型.

所以在这个地方数组与指针是不相同的,因为此时数组取值的第一步并不是从p中读取地址来再加上偏移量的.此时的p这个地址的值就为’h’这个字符,其类型为字符型而不是一个指针类型.此时p的地址与p[0]的地址是相同的.

同时,每个符号的地址在编译时就确定了,所以这里p的地址就已经确定了,如果需要访问p[2],则直接使用该地址加上2这个偏移量就可以取到这个值了.它不需要指令再取得这个首地址.而第一种情况下,还需要指令取得指针中的值,并通过这个值来访问字符串.

 

一个直观的看法就是前一种是两个不同的, 而后一种则是在同一个里面.

 

另外还有一个不同之处在于第一个情况下 p 指向的是一个常量区, 是不能改变的, 即不能够对p[i]赋值, 而第二种情况下, p是一个字符数组, 其是可以改变的. 可以对p[i]赋值的.

对于二维数组的一点分析
 
int a[10][20];
int **p;
p = a;
这种方式是错误的, 为什么?
因为a是一个数组类型, 而p是一个指向指针的指针类型, 此时p里面是一个地址, 通过该地址所得到的值还是一个整型指针的类型. 即*p也是一个整型指针. 而此处a为数组, 其里面的内容为整型, 让p指向这个数组的首地址, 而该首地址里面的内容又为一个整型,不是一个指针类型.所以会出现错误.
因此, 只能采取指向数组的指针类型才能够完成这个功能,即
int (*q)[20];
q = a;
此时q是一个指针,其指向的是一个具有20个整型变量的一维数组.
这样, 就可以使用q[i][j]或者*(*(q+i)+j)的方式来访问这个二维数组了.
另外, 对于一维数组.
int a[10];
int (*q)[10] = &a;
也可以进行访问, 但是此时q[i][j]中的i值就只能是为0了.
在这个地方, 也可以说明为什么a+1与&a+1不相同了. a表示的是一个元素, 所以a+1也只是地址增加一个整型的长度, 而&a+1则是以整个数组为角度来看的, 相当于q+1, 所以其地址增加的是10*sizeof(int)这么长.
同理,对于二维数组, int a[10][20]; a+1地址增加的为20个整型的长度. 因为其是以一个一维数组的角度来看的, 此时相当于是int (*q)[20] = a;这种形式.
a+1与*(a+1)都是同一个地址, 但是角度不相同, 前一个是以一个20长度的一维数组来看的, 即a[1]的地址, 而后面一个则是a[1]的值, 因为a[1]为a[1][0]的首地址, 而该首地址又与a[1]的首地址相同, 所以此时a+1与*(a+1)是相同的. *(a+1)+2就是a[1][2]的首地址了, 即相当于是a[1]+2了. 此时是以单个元素来看的.
阅读(685) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~