原创 C语言反省之一

2013-2-20 20:00 2463 8 19 分类: 消费电子

自己以前用结构体时,较少封装实现方法即:函数指针。这一点,造成了现在对于程序扩展起来有些捉襟见肘,这里以后要注意。

同时,函数参数也尽量使用结构体,这样避免函数接口的改动。
现在慢慢明白了,为什么数据结构在C语言中这么重要。一个数据结构的合适与否,影响了整个程序的框架以及以后的可扩展性。
 
以后使用C语言的过程要慢慢总结。
 
关于内存分配的再次理解:
最近看到,关于内存分配这个地方明白了,并不是我们的malloc分配多少内存,就开辟多少;也不是我们释放了多少内存,就释放了多少内存。
首先,在系统上,malloc有一个它自己的最小分配字节(这个在Linux系统上,一般是512字节)。为了free,还要留几个字节的内存保存开辟内存的信息(大小内存,一般是4个字节)。也就是使用malloc开辟内存后,我们得到的内存至少是512字节,我们使用的至少是4+开辟的字节大小。所以,malloc尽量别太多,否则,但是每次多余保留信息的四个字节就会占用不少内存。
 
由于malloc分配的内存,可能在不同的时间free释放,而且释放的顺序也可以不分先后。但是由于很多时候,系统回收内存都是从malloc最后分配的内存开始,如果最后分配的内存没有释放,很可能即使在它之前的内存,都已经释放了,由于最后的内存没有释放,造成系统无法回收内存。内存依然被程序占有。造成了内存空洞。所以,很多时候即使我们free了,内存也并没有归还给系统。
另外,如果释放分配内存的中间部分,再次用来分配时,不可能保证和释放时同样大小,没有被分配的内存在这以后可能因为较小,无法用来再分配,于是,这段内存就无法再被使用,造成了小的内存碎片。
这些小内存无释放内存被使用(即无法被用来分配使用)
 
关于越界访问的理解:
以前总以为越界访问主要造成的是,当前函数的操作出错。最近在看了《深入理解计算机系统》第三章程序的机器级表示后,发现自己理解错了。再结合,依稀还有的51单片机内容,发现越界访问,造成的将是整个程序的崩溃。如果是在单片机上,将是整个系统的崩溃(注1)。因为越界访问,很容易造成栈中,保存的信息被修改,造成主调函数的返回地址错误,当调用ret指令的时候,程序计数寄存器PC将被装入一个错误的返回地址,程序将跑飞。那没里内存错误也就不远了。现在编译器(注2)可能在里边加了从被调函数返回时的判断,如果发现编译器设置的栈中的信息被修改,程序将不返回,直接报错。
好一点的情况是,虽然越界,但是没有修改程序的返回信息。这样程序还是能返回,但是本函数的执行结果,将会有问题。
 
注:
1、单片机上的程序直接操作内存,如果越界访问,修改了栈的函数调用信息,将会造成程序不能从被调函数回到主调函数。程序计数器PC的值将指向不期望的地方。这样程序就跑飞了,这时看门狗就出来。或许这就是现在在单片机里加MPU的原因吧,防止直接操作内存时造成对内存的破坏。
2、现在的GCC是有了这样的功能。

文章评论11条评论)

登录后参与讨论

用户1586509 2013-3-13 08:23

有感触,谢谢分享

用户1590412 2013-3-11 16:15

学习了,谢谢分享

用户1138908 2013-3-9 10:12

学习学习。

1989tie_959541171 2013-3-2 19:50

你说的这个很对。 关于函数指针,让我们编程灵活了很多。 在工作中,发现51单片机对于函数指针的支持可能不太好,用的时候要留心;M3肯定没问题;其他架构的MCU不太清楚。

1989tie_959541171 2013-3-2 19:46

堆的情况,主要是由内存管理算法来处理。C没有什么规定。根据需要的情景,适合相应的内存管理算法。嵌入式编程,一般更多的预先分配好固定的内存再使用。这个和PC上边的分配有很大区别。主要是内存受限的原因。 所以一般嵌入式编程,不怎么使用malloc,即使有分配内存,也是自己实现的预先分配。 关于这个有一本《内存受限系统之软件开发》关于这方面总结的挺好,可以参考下。

用户1558497 2013-3-1 21:43

很好的解读

用户1318221 2013-3-1 13:07

我得反省为啥读起来那么累?

johnas_chai_229720659 2013-3-1 12:50

数据架构Data Structure 是软件编程基础的核心, C语言的函数指针Function pointer是最能体现编程架构的手段之一,现在他一语道破,直接点到重点。

用户1629079 2013-3-1 11:28

唔……没看懂。看着有点乱。楼主自身理解再整理一下咯。 越界分堆越界和栈越界,两种情况表现得不一样。楼主说了栈越界的情况,有兴趣再把堆的也聊下?

用户1670013 2013-3-1 08:42

看不懂也顶一个
相关推荐阅读
catch2000 2015-07-19 11:44
信号线小电阻的作用
在一块新的PCB上,测试系统能否正常运行的时候,发现系统上电后没有正常启动。  系统框图如下:   在上电的时刻,CPU A(GPIO电平2.6V)会向串口发送启动日志数据,CPU A启动后,...
catch2000 2015-07-05 17:04
协议设计中ACK机制的影响
在TCP/IP中,延时ACK和Nagle算法。  TCP为了同时处理成块数据(通常为512字节的用户数据)和交互数据(通常用户数据比较少,例如不大于10个字节),采用了延时ACK和Nagle算法...
catch2000 2015-05-23 15:48
话说物联网操作系统
最近好多家都宣布推出自己的物联网操作系统。   1. Google将要在Google I/O大会发布的Brillo; 2. 三星推出的Artik芯片搭载Mentor Graphics的...
catch2000 2015-03-31 23:52
不要采用异或来交换两个变量
在进行两个变量的时候,经常会看到有些书误人子弟的推荐使用异或的方式: 方式一 {   x = x ^ y;   y = x ^ y;   x = x ^ y; } 而不是...
catch2000 2014-10-09 07:28
为什么要测试先行
在产品的研发过程中,测试一项至关重要。不论是软件还是硬件。   软件的测试先行,在研发过程中,就做到质量的保证,因为在出现Bug的时候,容易定位Bug,而且即使是在客户端出现Bug,也能够...
catch2000 2014-10-09 07:26
C语言的面向对象编程(一)
一、前言 对于编程而言,重要的是解决问题的方式,而不是语言本身。面向对象与面向过程是解决问题和思考问题的方式。C语言常说是面向过程开发的语言,因为其缺少很多对于面向对象特性的支持。但,这并不影...
我要评论
11
8
关闭 站长推荐上一条 /2 下一条