C/C++——C语言数组名与指针
版权声明:原创文章,转载请注明出处。
1. 一维数组名与指针
对于一维数组来说,数组名就是指向该数组首地址的指针,对于:
int array[];
array就是该数组的首地址,如果我们想定义一个指向该数组的指针,我们可以用如下定义:
int *p;
然后令:
p = array;
那么指针p就指向了数组array的首地址,此时我们可以向操作array一样操作p:
#include <stdio.h>
int main(){
int array[]= {,,,,,,,,,};
int *p = array;
for(int i=;i<;i++){
printf("%d ",array[i]);
}
printf("\n");
for(int i=;i<;i++){
printf("%d ",p[i]);
}
printf("\n");
for(int i=;i<;i++){
printf("%d ",*(array+i));
}
printf("\n");
for(int i=;i<;i++){
printf("%d ",*(p+i));
}
printf("\n");
return ;
}
结果如下:

2. 多维数组名与指针
但是如果我们定义一个多维数组,如:
int arrays[2][2];
数组名arrays是数组元素的首地址,如果我们还用上面的那种定义:
int *p;
p = arrays;
这样p就指向了该多维数组的首地址,由于多维数组其实是数组的数组,所以多维数组的首地址就是一个地位的数组的首地址,即arrays[0],它是一个有2个元素的数组。因此我们可以将一维数组名赋值给int*型的指针,
#include <stdio.h>
int main(){
int arrays[][] = {,,,};
int *p = arrays;
for(int i=;i<;i++){
printf("%d ",p[i]);
}
printf("\n");
return ;
}
结果如下:

可以发现,编译时发生警告:

我们之所以可以通过p来遍历arrays中所有元素,是因为,数组本质上是按行优先存储的。
我们可以通过两次解引用获取到数组中的某一个元素的值,有下式成立:
**arrays == *&arrays[][] == arrays[][]
如:
#include <stdio.h>
int main(){
int arrays[][] = {,,,};
int *p = arrays;
printf("%d == %d == %d\n",**arrays,*&arrays[][],arrays[][]);
return ;
}
结果:

对于多维数组我们可以定义一个指向多维数组的指针:
int (*p)[2];
p = arrays;
这个括号是一定要加的,因为[]的优先级比较高(?)。
比较 int *p[2] 和 int (*p)[2] :
- int *p[2] 表示p是一个有2个元素的数组,数组中的每一个元素表示一个指向int型变量的指针;
- int (*p)[2] 表示p是一个指针,它指向一个数组,数组中的每个元素指向一个有2个元素的数组。
因此我们知道 int (*p)[2] 与arrays有同样的数组结构,p就是arrays的一个别名,我们可以通过p来实现任何arrays能实现的操作:
#include <stdio.h>
int main(){
int arrays[][] = {,,,,,};
int (*p)[] = arrays;
for(int i=;i<;i++){
for(int j=;j<;j++){
printf("%d ",p[i][j]);
}
printf("\n");
}
return ;
}
结果如下:

3. 数组名和指针的区别
数组名和指针之间,经常会交替使用这两个变量,例如把一个指针当成数组来使用,或者是把数组名赋值给指针,通过指针来访问数组成员变量。
但是,数组名和指针毕竟是定义不同的变量,它们之间也有一定的区别和联系。
理解数组名和指针的区别和联系有助于我们正确使用C语言,即什么情况下该使用数组名,什么情况下该使用指针。
区别1:
数组名和指针取地址后的值不一样:
- 数组名取地址得到的是数组名所指元素的地址。
- 对指针取地址得到的是指针变量自身的地址。
1)对于数组名:
#include <stdio.h>
int main(){
int array[] = {,,,,,};
printf("array = 0x%x\n",array);
printf("&array[0] = 0x%x\n",&array[]);
printf("&array = 0x%x\n",&array);
return ;
}
输出结果如下:

我们已经知道,数组名即数组首元素的地址,因此array和&array[0]的值一样,但是令我们惊奇的是,数组名的地址&array和和数组首元素的地址&array[0]也一样。
2)对于指针:
#include <stdio.h>
int main(){
int array[] = {,,,,,};
int *p = array;
printf("array = 0x%x\n",array);
printf("&array[0] = 0x%x\n",&array[]);
printf("&array = 0x%x\n",&array);
printf("\np = 0x%x\n",p);
printf("&p = 0x%x\n",&p);
return ;
}
结果如下:

可以发现指针的值和指针所在的地址是不同的。
区别2:
数组名是指针常量,指针是指针变量。
- 数组是固定大小的,数组一经定义,那么数组名就是一个指向数组首元素类型的常量指针,也就是说数组名是不允许更改的;
- 但是我们知道除非定义指针常量,否则指针变量是可以再赋值的。
区别3:
当对数组名使用sizeof时,得到的是数组所有元素所占的字节数,对指针sizeof得到的是指针类型的字节数。
#include <stdio.h>
int main(){
int array[] = {,,,,,};
int *p = array;
printf("sizeof array = %d bytes\n",sizeof(array));
printf("sizeof pointer = %d bytes\n",sizeof(p));
return ;
}
输出如下:

区别4:
对数组名取&和对指针取&的意义不同。
#include <stdio.h>
int main(){
int array[] = {,,,,,};
int *p = array;
printf("&array = 0x%x\n",&array);
printf("&array + 1 = 0x%x\n\n",&array + );
printf("&p = 0x%x\n",&p);
printf("&p + 1 = 0x%x\n",&p + );
return ;
}
输出如下:

上图可以看到对array取地址后加一,增大了24个字节,恰好是数组的大小;而对指针p取地址后加一,只增大了8个字节,恰好是一个指针类型所占的字节数。
4. 总结
- 数组名代表了一个指向数组首元素的指针常量,一经定义,不可更改;指针是指针变量,定义之后仍可更改,其类型在定义时确定。
- 当出现 sizeof 和 & 操作符时,数组名不再当成指向一个元素的指针常量来使用,而指针仍当成指向一个元素的指针变量来使用。
- 对于使用指针和数组下标的选择:
- 系统在使用数组下标对数组成员变量进行访问时,开销比较大,指针的访问效率是远远大于数组名的访问效率的;
- 但是只有在指针正确访问时,使用指针才比下标法更有效率;
- 下标法更加容易理解,在可读性方面,也更加的具有优势。
参考资料:
- https://blog.csdn.net/findgeneralgirl/article/details/78501734
- https://blog.csdn.net/dream_follower/article/details/80356754
C/C++——C语言数组名与指针的更多相关文章
- 对于C语言中数组名是指针的理解
我们都知道,c语言中数组名是一个指针,比如下面这段代码 #include<iostream>using namespace std;int main(){ int a[4]={1,2,3, ...
- C语言 数组名不是指针
今天上计算机系统课的时候老师讲到了C中的聚合类型的数据结构.在解释数组名的时候说"数组名是一个指针,指向该数组的第一个元素",附上ppt(第二行): 我觉得这是不正确的,是一个常见 ...
- C语言 数组名不是首地址指针
今天上计算机系统课的时候老师讲到了C中的聚合类型的数据结构.在解释数组名的时候说"数组名是一个指针,指向该数组的第一个元素",附上ppt(第二行): 我觉得这是不正确的,是一个常见 ...
- 【转】C语言中数组名和指针的区别
注:本文转自http://www.cnblogs.com/furaibo/archive/2010/03/19/1689710.html 魔幻数组名 请看程序(本文程序在WIN32平台下编译): #i ...
- 数组名和指针的深入理解(C++)
指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很多程序设计者就被搞糊涂了.魔幻数组名请看程序(本文程序在WIN32平台下编译): #includ ...
- c/c++数组名和指针区别深入探索
指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很多程序设计者就被搞糊涂了.而许多的大学老师,他们在C语言的教学过程中也错误得给学生讲解:&quo ...
- C/C++数组名与指针区别深入探索(转载)
转载自: C/C++数组名与指针区别深入探索 引言 指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很多程序设计者就被搞糊涂了.而许多的大学老师, ...
- CPP-基础:C/C++数组名与指针的区别
2005-08-23 08:36 来源:天极网 作者:宋宝华 责任编辑:方舟·yesky 引言 指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很 ...
- C/C++数组名与指针的区别详解
1.数组名不是指针我们看下面的示例: #include <iostream> int main() { ]; char *pStr = str; cout << sizeof( ...
随机推荐
- 【明哥报错簿】之【HTTP Status 500 - Servlet.init() for servlet mvc-dispatcher threw exception】
报错:java.lang.NoClassDefFoundError: /factory/config/EmbeddedValueResolver spring或者jdk的问题,解决办法:spring3 ...
- 洛谷 [USACO09OPEN]工作调度
题面 读完题,我们会发现有一个很重要的信息,每件物品代价相同,但价值不同.那么我们很容易想到,在满足限制的情况下,我们肯定会选择价值尽可能大的物品. 我们可否用背包来实现呢,答案是否定的,或者说我不会 ...
- 深入理解JVM一类加载器原理
我们知道我们编写的java代码,会经过编译器编译成字节码文件(class文件),再把字节码文件装载到JVM中,映射到各个内存区域中,我们的程序就可以在内存中运行了.那么字节码文件是怎样装载到JVM中的 ...
- QT 选择对话框简单示例
QT 选择对话框简单示例 部分代码: pDialog->addSeparator(); QAction *pmb2 = pDialog->addAction(QString::fromLo ...
- 【BZOJ1416/1498】【NOI2006】神奇的口袋(数论,概率)
[BZOJ1416/1498][NOI2006]神奇的口袋(数论,概率) 题面 BZOJ1416 BZOJ1498 洛谷 题面都是图片形式是什么鬼.. 题解 考虑以下性质 1.\(x[1],x[2]. ...
- unity3d点击屏幕选中物体
原文 http://blog.csdn.net/mycwq/article/details/19906335 前些天接触unity3d,想实现点击屏幕选中物体的功能.后来研究了下,实现原理就是检测从 ...
- 洛谷 P2900 [USACO08MAR]土地征用Land Acquisition 解题报告
P2900 [USACO08MAR]土地征用Land Acquisition 题目描述 约翰准备扩大他的农场,眼前他正在考虑购买N块长方形的土地.如果约翰单买一块土 地,价格就是土地的面积.但他可以选 ...
- Kerberos的黄金票据详解
0x01黄金票据的原理和条件 黄金票据是伪造票据授予票据(TGT),也被称为认证票据.如下图所示,与域控制器没有AS-REQ或AS-REP(步骤1和2)通信.由于黄金票据是伪造的TGT,它作为TGS- ...
- Mac下安装MacProt,并GNU autotools的安装和使用 autoconf,automake
1 MacPort的下载:http://www.macports.org/install.php, 需要安装xCode支持macport 2 安装MacPorts 与其他Mac的软件的安装方式相同,挂 ...
- python基础----再看property、描述符(__get__,__set__,__delete__)
一.再看property 一个静态属性property ...