本文目录

前面我们已经学习了指针,如果指针存储了某个变量的地址,我们就可以说指针指向这个变量。数组及其数组元素都占有存储空间,都有自己的地址,因此指针变量可以指向整个数组,也可以指向数组元素。

一、用指针指向一维数组的元素

 1 // 定义一个int类型的数组
2 int a[2];
3
4 // 定义一个int类型的指针
5 int *p;
6
7 // 让指针指向数组的第0个元素
8 p = &a[0];
9
10 // 修改所指向元素的值
11 *p = 10;
12
13 // 打印第一个元素的值
14 printf("a[0] = %d", a[0]);

输出结果:,说明已经通过指针间接修改了数组元素的值,跟指向一个普通int类型变量是一样的。

由于数组名代表着数组的首地址,即a == &a[0],因此第8行代码等价于:

// 让指针指向数组的第0个元素
p = a;

内存分析图如下,一个指针变量占用2个字节,一个int类型的数组元素占用2个字节

二、用指针遍历数组元素

1.最普通的遍历方式是用数组下标来遍历元素

1 // 定义一个int类型的数组
2 int a[4] = {1, 2, 3, 4};
3
4 int i;
5 for (i = 0; i < 4; i++) {
6 printf("a[%d] = %d \n", i, a[i]);
7 }

输出结果:

2.接下来我们用指针来遍历数组元素

先定义一个指针,指向数组的第一个元素

// 定义一个int类型的数组
int a[4] = {1, 2, 3, 4}; // 定义一个int类型的指针,并指向数组的第0个元素
int *p = a;

p的值是a[0]的地址,因此,现在我们利用指针 p只能访问数组的第0个元素a[0],用*p就可取出a[0]的值1。要想访问其他元素,就必须拿到元素的地址,可以发现每个元素的地址差值为2,因为在 16位编译器环境下,一个int类型的变量占用2个字节。现在只是知道a[0]的地址值为p,怎么根据a[0]的地址获取其他元素的地址呢?其实非常简 单,p+1就是a[1]的地址。注意了,这里的p+1代表着p的值加2,并不是p的值加1,比如p的值为ffc3,p+1则为ffc5,而非ffc4。依 次类推,p+2就是a[2]的地址ffc7,p+3就是a[3]的地址ffc9。

我先解释一下,为什么p+1代表p的值加2,而不是加1呢?

其实,p+1不一定代表p的值加2,也可能是加1、加4或者加8。究竟加多少,这跟指针的类型有关。下图是在16位编译器环境下的情况。

聪明的你可能已经找到规律了,因为char类型的变量要占用1字节,所以p+1代表p的值加1;float类型的变量占用4字节,所以p+1代表p的值加4。从这一点,也可以很好地说明为什么指针一定要分类型,不同类型的指针,p+1的含义是不一样的。

上述代码中的p指向了int类型的数组元素a[0],所以p+1代表p的值加2。知道怎么获取其他元素的地址了,那么就可以利用指针p遍历数组元素了。

 1 // 定义一个int类型的数组
2 int a[4] = {1, 2, 3, 4};
3
4 // 定义一个int类型的指针,并指向数组的第0个元素
5 int *p = a;
6
7 int i;
8 for (i = 0; i < 4; i++) {
9 // 利用指针运算符*取出数组元素的值
10 int value = *(p+i);
11
12 printf("a[%d] = %d \n", i, value);
13 }

注意第10行的代码,*(p+i)代表根据p+i的值(其实就是第i个数组元素的地址)访问对应的存储空间,并取出存储的内容(也就是取出第i个数组元素的值),赋值给左边的value。

最后的输出效果是一样的:。注意的是:遍历完毕后,指针变量p还是指向a[0],因为p值一直没有变过,一直都是a[0]的地址ffc3。

补充一下,其实第10行改成下面的代码也是可以的:

int value = *(a+i);

大家都知道,a值代表数组的首地址,也就是a[0]的地址ffc3。a+1则代表a的值加2,即a[1]的地址ffc5,也就是说,a+i代表着元素a[i]的地址。相信大家也能猜出来了,a+1不一定代表着a值加2,究竟加多少,取决于数组的类型。a+i的计算方法与p+i相同。

利用上面的方法遍历完数组元素后,p一直指向元素a[0]。其实我们也可以直接修改p的值来访问数组元素,只需要改一下第10行的代码即可

// 利用指针运算符*取出数组元素的值
int value = *(p++);

p++其实就是相当于p = p + 1,直接修改了p值,而且每次是加2。因此,每执行一次p++,指针p就会指向下一个数组元素。

输出结果肯定是一样的:。但是,遍历完毕后,指针变量p没有指向任何数组元素,因为一共执行了4次p++,最后p值为ffcb。当然,可以重新让p指向a[0]:p = &a[0];或者p = a;

注意,这里的写法是错误的

int value = *(a++);

a++相当于a=a+1,数组名a是个常量!不能进行赋值运算!

三、指针与数组的总结

p是指针,a是一个数组

1> 如果p指向了一个数组元素,则p+1表示指向数组该元素的下一个元素。比如,假设p = &a[0],则p+1表示a[1]的地址

2> 对于不同类型的数组元素,p值的改变是不同的。如果数组元素为int类型,p+1代表着p的值加上2(16位编译器环境下)

3> 如果p的初值是&a[0],那么

  • p+i和a+i都可以表示元素a[i]的地址,它们都指向数组的第i个元素。a代表数组首地址,a+i也是地址,它的计算方法与p+i相同
  • *(p+i)和*(a+i)都表示数组元素a[i]
  • 虽然p+i和a+i都指向数组的第i个元素,但二者使用时还是有区别的。因为作为指针变量的p可以改变自身值,如p++,使p的值自增。而数组名a是一个代表数组首地址的常量,它的值是不能改变的,即a++是不合法的

4> 引用一个数组元素可以有两种方法:

  • 下标法: 如a[i]
  • 指针法: 如*(p+i) 或 *(a+i)

四、数组、指针与函数参数

1.用数组名作为函数实参时,是把实参数组的首地址传递给形参数组,两个数组共同占用同一段内存空间,这样形参数组中的元素值发生变化就会使实参数组的元素值也同时变化

 1 void change(int b[]) {
2 b[0] = 10;
3 }
4
5 int main()
6 {
7 // 定义一个int类型的数组
8 int a[4] = {1, 2, 3, 4};
9
10 // 将数组名a传入change函数中
11 change(a);
12
13 // 查看a[0]
14 printf("a[0]=%d", a[0]);
15
16 return 0;
17 }

change函数的形参是数组类型的,在第11行调用change函数时,将数组名a,也就是数组的地址传给了数组b。因此数组a和b占用着同一块内存空间。

输出结果:

2.这种地址的传递也可以用指针来实现。函数的实参和形参都可以分别使用数组或指针。这样就有4种情况:

也就是说,如果一个函数的形参类型是一个数组,调用函数时,你可以传入数组名或者指针变量;

 1 void change(int b[]) {
2 b[0] = 10;
3 }
4
5 int main()
6 {
7 // 定义一个int类型的数组
8 int a[4] = {1, 2, 3, 4};
9
10 int *p = a;
11
12 // 将数组名a传入change函数中
13 change(p);
14
15 // 查看a[0]
16 printf("a[0]=%d", a[0]);
17
18 return 0;
19 }

注意第1行的形参类型是个数组int b[],第10行定义了指针变量p,第13行将p当做实参传入函数

如果一个函数的形参类型是一个指针变量,调用函数时,你可以传入数组名或者指针变量。

 1 void change(int *b) {
2 b[0] = 10;
3 // 或者*b = 10;
4 }
5
6 int main()
7 {
8 // 定义一个int类型的数组
9 int a[4] = {1, 2, 3, 4};
10
11 // 将数组名a传入change函数中
12 change(a);
13
14 // 查看a[0]
15 printf("a[0]=%d", a[0]);
16
17 return 0;
18 }

注意第1行的形参类型是个指针变量int *b,第12行将数组名a当做实参传入函数。

由第2行可以看出,在很多情况下,指针和数组是可以相互切换使用的。但是,并不能说指针就等于数组。

【C语言】指向一维数组元素的指针的更多相关文章

  1. 【C语言】-指向一维数组元素的指针

    本文目录 一.用指针指向一维数组的元素 二.用指针遍历数组元素 三.指针与数组的总结 四.数组.指针与函数参数 说明:这个C语言专题,是学习iOS开发的前奏.也为了让有面向对象语言开发经验的程序员,能 ...

  2. 【C语言】12-指向一维数组元素的指针

    一.用指针指向一维数组的元素 1 // 定义一个int类型的数组 2 int a[2]; 3 4 // 定义一个int类型的指针 5 int *p; 6 7 // 让指针指向数组的第0个元素 8 p ...

  3. [C++程序设计]指向数组元素的指针

    如果先使p指向数组a的首元素(即p=a),则: (1) p++(或p+=1).使p指向下一元素,即a[1]. 如果用*p,得到下一个元素a[1]的值. (2) *p++.由于++和*同优先级,结合方向 ...

  4. 【C语言】-一维数组

    数组: 数组是一系列相同类型的有序数据的集合,数组中的每一个元素都是同一个数据类型,所有元素共用一个名字,用下标来区别数组中的每一个元素. C语言中,数组属于构造数据类型.一个数组中含有多个数组元素, ...

  5. C语言之一维数组与指针

    一维数组: 假如有一维数组如下: ]; 该数组有3个元素,数据类型为char型,地址空间如下. 如果想访问数据,直接使用a[0].a[1].a[2]取出相应地址空间的值即可 一级指针: 指针即地址,c ...

  6. c语言操作一维数组-3

    C语言选择题#includemain(){double a[15],k;k=fun(a);} 则以下选项中错误的fun函数首部是 ( D)A.double fun(double a[15]) B.do ...

  7. C语言学习笔记--数组参数和指针参数

    1. 数组参数退化为指针的意义 (1)C 语言中只会以值拷贝的方式传递参数,当向函数传递数组时,将整个数组拷贝一份传入函数导致执行效率低下,C 语言以高效作是最初的设计目标,所以这种方法是不可取的. ...

  8. [转] C语言多维数组与多级指针

    http://c.biancheng.net/cpp/html/477.html 多维数组与多级指针也是初学者感觉迷糊的一个地方.超过二维的数组和超过二级的指针其实并不多用.如果能弄明白二维数组与二级 ...

  9. C语言中字符数组和字符串指针分析

    这几天搞Unix上的C程序,里面用到了很多字符数组和字符串指针,我记得在学完C语言后相当一段时间里,对指针这个东西还是模模糊糊,后来工作也没怎么 用到过C,虽然网上这类的文章也有很多,还是决定自己在这 ...

随机推荐

  1. 根据json对象的值替换json数组里的值

    功能: var fruitArry=[{name:'durian'},{name:'peach'},{name:'banana'},{name:'pitaya'},{name:'apple'},{na ...

  2. tornado 数据库操作

    tornado是python的web框架,web程序开发中数据库操作是必须的. 安装: tornado的官方文档中提供的说明比较少,而且提供的模块中未找到数据库方面的模块,难道没有针对数据库操作进行封 ...

  3. mysql 外键 cascade

    1 . cascade方式在父表上update/delete记录时,同步update/delete掉子表的匹配记录 2. set null方式在父表上update/delete记录时,将子表上匹配记录 ...

  4. SpringBoot应用配置常用相关视图解析器

    目录 SpringBoot的自动装配装配了视图解析器了吗? SpringBoot使用JSP SpringBoot中使用Thymeleaf SpringBoot中使用Freemark SpringBoo ...

  5. HTTP协议的前世今生——各版本HTTP协议对比

    HTTP协议是如今互联网与服务端技术的基石,HTTP协议的演进也从侧面反应了互联网技术的快速发展.这两天在准备一次关于HTTP1.1协议特性的技术分享过程中,顺便了解了下各版本HTTP协议的特点,在这 ...

  6. ELk之使用kibana展示访问IP地图

    参考文档:http://blog.51cto.com/ls40905250/1915280 https://blog.csdn.net/zsjwish/article/details/79792212 ...

  7. HDU 2476 String painter(区间DP)

    String painter Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...

  8. PAT甲1005 Spell it right【字符串】

    1005 Spell It Right (20 分) Given a non-negative integer N, your task is to compute the sum of all th ...

  9. POJ 2912 - Rochambeau - [暴力枚举+带权并查集]

    题目链接:http://poj.org/problem?id=2912 Time Limit: 5000MS Memory Limit: 65536K Description N children a ...

  10. codeforces 894B - Ralph And His Magic Field - [数学题]

    题目链接:https://cn.vjudge.net/problem/CodeForces-894B Ralph has a magic field which is divided into n × ...