`
mocca3in1
  • 浏览: 44206 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

c语言内存使用

阅读更多
文/楚云风  出处/天极网

  问题:内存使用

  有人写了一个将整数转换为字符串的函数:

char *itoa (int n)
{
  char retbuf[20];
  sprintf(retbuf, "%d", n);
  return retbuf;
}

  如果我调用这个函数:char *str5 = itoa(5),str5会是什么结果呢?

  答案分析:

  答案是不确定,可以确定的是肯定不是我们想要的 “5”。

   retbuf定义在函数体中,是一个局部变量,它的内存空间位于栈(stack)中的某个位置,其作用范围也仅限于在itoa()这个函数中。当 itoa()函数退出时,retbuf在调用栈中的内容将被收回,这时,这块内存地址可能存放别的内容。因此将retbuf这个局部变量返回给调用者是达不到预期的目的的。

  那么如何解决这个问题呢,不用担心,方法不但有,而且还不止一个,下面就来阐述三种能解决这个问题的办法:

  1)、在itoa()函数内部定义一个static char retbuf[20],根据静态变量的特性,我们知道,这可以保证函数返回后retbuf的空间不会被收回,原因是函数内的静态变量并不是放在栈中,而是放在程序中一个叫“.bss”段的地方,这个地方的内容是不会因为函数退出而被收回的。

  这种办法确实能解决问题,但是这种办法同时也导致了itoa()函数变成了一个不可重入的函数(即不能保证相同的输入肯定有相同的输出),另外, retbuf [] 中的内容会被下一次的调用结果所替代,这种办法不值得推荐。

  2)、在itoa()函数内部用malloc() 为retbuf申请内存,并将结果存放其中,然后将retbuf返回给调用者。由于此时retbuf位于堆(heap)中,也不会随着函数返回而释放,因此可以达到我们的目的。

  但是有这样一种情况需要注意:itoa()函数的调用者在不需要retbuf的时候必须把它释放,否则就造成内存泄漏了,如果此函数和调用函数都是同一个人所写,问题不大,但如果不是,则比较容易会疏漏此释放内存的操作。

  3)、将函数定义为 char *itoa(int n, char *retbuf),且retbuf的空间由调用者申请和释放,itoa()只是将转换结果存放到retbuf而已。

  这种办法明显比第一、二种方法要好,既避免了方法1对函数的影响,也避免了方法2对内存分配释放的影响,是目前一种比较通行的做法。

   4)参见《C++沉思录》,还有一种方法是:类似方法2,但是调用者不需要释放。retbuf所指向的内存块的有效期保存到下一次itoa()函数调用为止。但是,如果调用者需要保留原来的结果,需要复制一份。

  扩展分析:

  其实就这个问题本身而言,我想大家都可以立刻想到答案,关键在于对内存这种敏感资源的正确和合理地利用,下面对内存做一个简单的分析:

  1)、程序中有不同的内存段,包括:

  .data - 已初始化全局/静态变量,在整个软件执行过程中有效;

   .bss - 未初始化全局/静态变量,在整个软件执行过程中有效;

  .stack - 函数调用栈,其中的内容在函数执行期间有效,并由编译器负责分配和收回;

  .heap - 堆,由程序显式分配和收回,如果不收回就是内存泄漏。

  2)、自己使用的内存最好还是自己申请和释放。

  这可以说是一个内存分配和释放的原则,比如说上面解决办法的第二种,由itoa()分配的内存,最后由调用者释放,就不是一个很好的办法,还不如用第三种,由调用者自己申请和释放。另外这个原则还有一层意思是说:如果你要使用一个指针,最好先确信它已经指向合法内存区了,如果没有就得自己分配,要不就是非法指针访问。很多程序的致命错误都是访问一个没有指向合法内存区的指针,这也包括空指针。
分享到:
评论
1 楼 shaxquan 2010-06-17  
第三种情况,调用者很容易忘记给retbuf分配空间,初学者很容易这样使用strcpy,
int main() {
     char *str;
     strcpy(str, "hello,world");
     printf(str);

     return 0;
}
strcpy的函数原型很容易让人误解,char *strcpy(char *dest, const char *src);
误以为strcpy只需传入两个char指针.

相关推荐

    C语言内存使用详解.pdf

    C语言内存使用详解.pdf C语言内存使用详解.pdf C语言内存使用详解.pdf

    C语言内存使用详解

    学习C语言,最重要的莫过于内存访问方式!

    C语言内存精讲,让你彻底明白C语言的运行机制!

    4. C语言内存对齐,提高寻址效率 5. 内存分页机制,完成虚拟地址的映射 6. 分页机制究竟是如何实现的? 7. MMU部件以及对内存权限的控制 8. Linux下C语言程序的内存布局(内存模型) 9. Windows下C语言程序的内存...

    C语言内存精讲.rar

    理解指针的工作原理,包括指针的声明、使用和指针运算,是掌握C语言内存管理的关键。 6. **内存泄漏检测**:为了找出程序中的内存泄漏,可以使用特定的工具如Valgrind,或者编写特定的代码片段来检查内存分配和释放...

    C语言内存管理

    C语言内存管理是一个程序员必须掌握的重要知识点,因为它直接关系到程序运行的稳定性和效率。内存泄漏是C语言中一个非常典型的问题,指的是程序中分配的堆内存无法再被访问或回收,最终导致可用内存不断减少。内存...

    (麻省理工免费课程)C语言内存管理和C++面向对象编程

    ### 麻省理工免费课程:C语言内存管理和C++面向对象编程 #### 一、C语言简介 在《麻省理工免费课程:C语言内存管理和C++面向对象编程》这一课程中,首先对C语言进行了介绍。C语言是一种历史悠久且功能强大的编程语言...

    C语言内存模型总结

    对c语言内存模型和粗浅认识和总结,我们对一门学问的认识都有一个从散落到集中过程,权当一次小小的总结吧。

    推荐4款linux下的c语言内存泄漏检测工具.zip

    推荐4款linux下的检测c语言编写的程序的内存泄漏工具 C语言和其他语言相比最大的特色就是能够操作内存 但是最常犯的错误也是内存泄漏(管杀不管埋) 所以我们需要用一些工具来帮助我们检测是否存在内存泄漏,存在多少 ...

    C语言内存管理资料

    有关C语言内存管理方面的培训资料,有兴趣的可以看下!

    存储引擎(C语言版本)_C语言内存管理_C语言内核编程_c语言数据结构_C语言存储引擎_

    首先,我们要理解C语言内存管理。在C语言中,程序员需要手动分配和释放内存,使用`malloc()`和`free()`函数来动态地申请和释放内存块。这样的方式虽然需要更多的编程技巧,但也提供了更高的灵活性和对内存使用的精确...

    C语言中级教程-6 内存使用

    C语言中级教程-6 内存使用

    C语言内存安全性运行时验证技术综述.pdf

    "C语言内存安全性运行时验证技术综述" ...C语言内存安全性运行时验证技术是软件开发中不可或缺的一部分,通过使用 AddressSanitizer、Move 等检测工具,可以快速检测和解决内存安全性错误,提高软件的可靠性和安全性。

    有关c语言内存的思考

    "c语言内存管理和内存分配" 在C语言中,内存管理和内存分配是非常重要的知识点。本文将通过一些思考题,来帮助读者更好地理解C语言中的内存管理和内存分配。 首先,让我们来看第一个问题。在这个问题中,我们定义...

    c语言内存泄露的问题

    讲述了c语言中导致内存泄露的原因及内存泄露的分类。

    收集一些C语言内存相关的项目.zip

    这个压缩包“收集一些C语言内存相关的项目.zip”可能包含了一系列的示例项目,旨在帮助开发者深入理解C语言中的内存操作。虽然没有具体的标签提供额外信息,但我们可以从标题和描述推测其主要内容可能包括动态内存...

    嵌入式C语言内存操作编程

    本篇主要讲述了嵌入式系统C 编程中内存操作的相关技巧.掌握并深入理解关于数据指针、函数指针、动态申请内存、const 及volatile ...我们之所以在嵌入式系统中使用C 语言进行程序设计,99%是因为其强大的内存操作能力!

    C语言内存,总结关于C语言内存的知识点

    C语言学习

    C语言内存操作的21份资料.

    理解指针的使用,包括指针的声明、赋值、解引用和传递,是掌握C语言内存操作的基础。 这些资料可能还会涉及以下主题: - 内存泄漏检测:如何查找和修复程序中的内存泄漏,如使用Valgrind等工具。 - 内存安全:了解...

    C语言内存管理详解---c语言雷区

    C语言内存管理是程序员必须掌握的重要技能,尤其是在使用C语言编程时,因为C语言提供了直接操控内存的机制。内存管理包括了内存的分配、使用和释放,如果不谨慎处理,可能会导致各种问题,如内存泄漏、访问越界、空...

Global site tag (gtag.js) - Google Analytics