C语言数组和指针的理解_在取地址运算上的操作_指针加减操作_a 和&a 的区别

简介:

1.一个实例+理论分析

 

在了解数组和指针的访问方式前提下,下面再看这个例子:

1
2
3
4
5
6
main()
{
int  a[ 5 ]={ 1 , 2 , 3 , 4 , 5 };
int  *ptr=( int  *)(&a+ 1 );
printf( "%d,%d" ,*(a+ 1 ),*(ptr- 1 ));
}

打印出来的值为多少呢? 这里主要是考查关于指针加减操作的理解


  对指针进行加1操作,得到的是下一个元素的地址,而不是原有地址值直接加1。所 以,一个类型为T的指针的移动,以sizeof(T) 为移动单位。

  因此,对上题来说,a是一个一维数组,数组中有5个元素,所以a的类型是数组指针;ptr是一个int 型的指针,ptr的类型是整型指针。 

1
&a +  1 :取数组a 的首地址,该地址的值加上sizeof(a) 的值,即&a +  5 *sizeof( int ),也就是下一个数组的首地址。  

  显然当前指针已经越过了数组的界限。


 (int *)(&a+1): 则是把上一步计算出来的地址,强制转换为int * 类型,赋值给ptr。

  

  *(a+1): a,&a的值是一样的,但意思不一样a是数组首元素的首地址,也就是a[0]的首地址,&a是数组的首地址,a+1是数组下一元素的首地址,即a[1]的首地址,&a+1是下一个数组的首地址。

  所以输出2*(ptr-1):因为ptr是指向a[5],并且ptr是int * 类型,所以*(ptr-1)是指向a[4],输出5。

 

2.Visual C++6.0上的真实调试结果


这些分析我相信大家都能理解,但是在授课时,学生向我(陈正冲老师)提出了如下问题:


在Visual C++6.0的Watch窗口中&a+1的值怎么会是(x0012ff6d(0x0012ff6c+1)呢?

 

上图是在Visual C++6.0调试本函数时的截图。

1
2
3
a在这里代表是的数组首元素的地址即a[ 0 ]的首地址,其值为 0x0012ff6c
&a代表的是数组的首地址,其值为 0x0012ff6c
a+ 1 的值是 0x0012ff6c + 1 *sizeof( int ),等于 0x0012ff70 。 


问题就是&a+1的值怎么会是(x0012ff6d(0x0012ff6c+1)呢?


按照我们上面的分析应该为0x0012ff6c+5*sizeof(int)。其实很好理解。当你把&a+1放到Watch窗口中观察其值时,表达式&a+1已经脱离其上下文环境,编译器就很简单的把它解析为&a的值然后加上1byte。而a+1的解析就正确,我(陈正冲老师)认为这是Visual C++6.0的一个bug。既然如此,我们怎么证明证明&a+1的值确实为0x0012ff6c+5*sizeof(int)呢?

  很好办,用printf函数打印出来。这就是我在本书前言里所说的,有的时候我们确实需要printf函数才能解决问题。你可以试试用printf("%x",&a+1);打印其值,看是否为0x0012ff6c+5*sizeof(int)。注意如果你用的是printf("%d",&a+1);打印,那你必须在十进制和十六进制之间换算一下,不要冤枉了编译器。

  另外我(陈正冲老师)要强调一点:不到非不得已,尽量别使用printf函数,它会使你养成只看结果不问为什么的习惯。比如这个列子,*(a+1)和*(ptr-1)的值完全可以通过Watch窗口来查看。平时初学者很喜欢用“printf("%d,%d",*(a+1),*(ptr-1));”这类的表达式来直接打印出值,如果发现值是正确的就欢天喜地。这个时候往往认为自己的代码没有问题,根本就不去查看其变量的值,更别说是内存和寄存器的值了。(嗯,这个坏习惯,我是有的。)

 

 

3. 最好不要利用printf函数进行调试

 

陈正冲老师的经验与教诲:

 

  更有甚者,printf函数打印出来的值不正确,就措手无策,举手问“老师,我这里为什么不对啊?”。长此以往就养成了很不好的习惯,只看结果,不重调试。这就是为什么同样的几年经验,有的人水平很高,而有的人水平却很低。其根本原因就在于此,往往被一些表面现象所迷惑。printf函数打印出来的值是对的就能说明你的代码一定没问题吗?我看未必。曾经一个学生,我让其实现直接插入排序算法。很快他把函数写完了,把值用printf函数打印出来给我看。我看其代码却发现他使用的算法本质上其实是冒泡排序,只是写得像直接插入排序罢了。等等这种情况数都数不过来,往往犯了错误还以为自己是对的。所以我平时上课之前往往会强调,不到非不得已,不允许使用printf函数,而要自己去查看变量和内存的值。学生的这种不好的习惯也与目前市面上的教材、参考书有关,这些书甚至花大篇幅来介绍scanf和printf这类的函数,却几乎不讲解调试技术。甚至有的书还在讲TruboC 2.0之类的调试器!如此教材教出来的学生质量可想而知。

 本文转自二郎三郎博客园博客,原文链接:http://www.cnblogs.com/haore147/p/3647231.html,如需转载请自行联系原作者

相关文章
|
10月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
10月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。
|
10月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
10月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
10月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
194 4
|
9月前
|
存储 数据可视化 C++
第九问:能否尽可能详细阐述指针和引用的区别?
在C++中,指针和引用是两个重要的概念,用于操作内存地址和数据。指针是一个存储内存地址的变量,可以动态分配和释放内存;引用是变量的别名,绑定后不可改变指向。指针提供更大的灵活性和控制力,适用于复杂内存操作;引用更直观,适合简化代码并提高可读性。根据实际需求选择合适的工具。
176 0
|
C语言
《C语言程序设计进阶教程》一2.3.6 获取地址
本文讲的是C语言程序设计进阶教程一2.3.6 获取地址,本节书摘来华章计算机《C语言程序设计进阶教程》一书中的第2章,第2.3.6节, Intermediate C Programming[美] 陆永祥(Yung-Hsiang Lu) 著 徐东 译 译更多章节内容可以访问云栖社区“华章计算机”公众号查看。
1111 0
|
2月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
236 15
|
8月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
369 23
|
7月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
176 1
一文彻底搞清楚C语言的函数