C与C++的内存机制的比较

简介:

C语言与C++内存非常相似,这也是我一直搞不清楚他的原因;下面梳理一下他们之间的区别:

  1、先说C语言的内存机制

    • 栈        位于函数内的局部变量(包括函数实参),由编译器负责分配和释放,函数结束,栈变量失效;

    • 堆        由程序员用malloc()/calloc()/realloc()分配空间,free()释放所申请的空间。如果程序员忘记free(),则会造成内存泄漏,程序结束时可能会由操作系统回收,也许就一直占用着直至关机。

    • 全局区/静态区        全局变量和静态变量存放区,程序一经编译好,该区域便存在。并且在C语言中初始化的全局变量和静态变量和未初始化的放在相邻的两个区域在C++中,由于编译器会给全局变量和静态变量自动初始化赋值,所以没有区分了)。由于全局变量一直占据内存空间且不易维护,推荐少用。程序结束时释放。 

    • C风格字符串常量存储区        专门存放字符串常量的地方,程序结束时释放;

    • 程序代码区         存放程序二进制代码的区域。


  2、再说C++的内存机制

    • 栈        位于函数内的局部变量(包括函数实参),由编译器负责分配释放,函数结束,栈变量失效。

    • 堆        这里与C不同的是,该堆是由new申请的内存,由delete负责释放。

    • 自由存储区        由程序员用malloc()/calloc()/realloc()分配空间,由free()释放。如果程序员忘记free()了,则会造成内存泄漏,程序结束时可能会有操作系统回收,也许就一直占用着直至关机。 与C的堆机制对应。

    • 全局区/静态区        全局变量和静态变量存放区,程序一经编译好,该区域便存在。在C++中,由于编译器会给全局变量和静态变量自动初始化赋值,所以没有区分初始化和未初始化变量。由于全局变量一直占据内存空间且不易维护,推荐少用。程序结束时释放。

    • 常量存储区        这是一块比较特殊的存储区,专门存储不能修改的常量(如果采用非正常手段更改,当然也是可以的)。



下面举个栗子,比较C与C++在全局区/静态区的区别(Linux):

1
2
3
4
5
6
7
8
9
10
#include <stdio.h>
 
static  int  a;
int  b;
int  c = 1;
 
int  main( void )
{
     return  0;
}

将文件编译成可执行文件,打印文件的大小:

1.PNG

可以看到,数据段(data,不包括bss)为252,bss(未初始化数据段)为16;

接下来将测试代码进行修改(对b进行初始化):

1
2
3
4
5
6
7
8
9
10
#include <stdio.h>
 
static  int  a;
int  b = 1;
int  c = 1;
 
int  main( void )
{
     return  0;
}

将文件编译成可执行文件,打印文件的大小:

2.PNG

可以看到,数据段(data,不包括bss)为256(252+4),bss(未初始化数据段)为12(16-4);


与上面的那段代码对比可以发现,data增加了4,刚刚好是bss减少的4。现在,就可以确定C语言中,对全局区/静态区中变量初始化与为初始化是放在不同区域的。

接下来看一下C++运行的结果:

b未手动初始化(int b;):

3.PNG

b手动初始化为0(int b = 0):

5.PNG

上面在C++下面,数据段(data)没有变化;因此在c语言中,全局变量又分为初始化的和未初始化的,在c++里面没有这个区分了,他们共同占用同一块内存区。


但是,这里面有一个非常小的细节对于b的初始化,如果手动初始化为0以外的数字,打印出来的data段又与C语言是一样的:

b初始化为1:

8.PNG

b不初始化:

7.PNG


以上问题的出现,是什么原因,还没有查清楚。

可能的原因是:

BSS段(bss segment)通常是指用来存放程序中未初始化的全局变量的一块内存区域。(0也算未初始化)
数据段(data segment)通常是指用来存放程序中已初始化且不为0的全局变量的一块内存区域。(只有初始化为0之外的数字才算真正的初始化)

具体原因不知道,只是猜测。



本文转自 七十七快 51CTO博客,原文链接:http://blog.51cto.com/10324228/2083826

相关文章
|
1月前
|
安全 算法 程序员
【C/C++ 文件操作】深入理解C语言中的文件锁定机制
【C/C++ 文件操作】深入理解C语言中的文件锁定机制
34 0
|
29天前
|
存储 缓存 Java
金石原创 |【JVM盲点补漏系列】「并发编程的难题和挑战」深入理解JMM及JVM内存模型知识体系机制(1)
金石原创 |【JVM盲点补漏系列】「并发编程的难题和挑战」深入理解JMM及JVM内存模型知识体系机制(1)
37 1
|
1月前
|
安全 算法 C++
【C++ 异常 】深入了解C++ 异常机制中的 terminate()处理 避免不必要的错误(三)
【C++ 异常 】深入了解C++ 异常机制中的 terminate()处理 避免不必要的错误
47 0
|
3天前
|
存储 算法
深入理解操作系统的内存管理机制
【4月更文挑战第24天】 在现代计算机系统中,操作系统扮演着资源管理者的角色,其中内存管理是其核心职责之一。本文将探讨操作系统如何通过内存管理提升系统性能和稳定性,包括物理内存与虚拟内存的概念、分页机制、内存分配策略以及内存交换技术。我们将透过理论与实践的结合,分析内存管理的关键技术及其对系统运行效率的影响。
|
4天前
|
存储 人工智能 程序员
【重学C++】【内存】关于C++内存分区,你可能忽视的那些细节
【重学C++】【内存】关于C++内存分区,你可能忽视的那些细节
35 1
|
10天前
|
存储 算法 数据安全/隐私保护
深入理解操作系统的内存管理机制
【4月更文挑战第17天】 在现代计算机系统中,操作系统扮演着资源管理者的角色,其中内存管理是其核心职能之一。本文探讨了操作系统内存管理的关键技术,包括虚拟内存、物理内存分配与回收、分页和分段机制,以及内存交换技术。通过分析这些机制的原理和实现,我们旨在加深读者对操作系统如何有效管理和保护内存资源的理解。
11 1
|
12天前
|
算法
深入理解操作系统的内存管理机制
【4月更文挑战第15天】 本文将探讨操作系统中至关重要的一环——内存管理。不同于通常对内存管理概念的浅尝辄止,我们将深入研究其核心原理与实现策略,并剖析其对系统性能和稳定性的影响。文章将详细阐述分页系统、分段技术以及它们在现代操作系统中的应用,同时比较它们的效率与复杂性。通过本文,读者将获得对操作系统内存管理深层次工作机制的洞见,以及对设计高效、稳定内存管理系统的理解。
|
22天前
|
缓存 监控 算法
深入理解操作系统的内存管理机制
【4月更文挑战第5天】 随着现代计算机系统的发展,操作系统的内存管理已成为确保系统高效稳定运行的关键因素。本文旨在探讨操作系统中内存管理的基本原理、关键技术及其在实际应用中的优化策略。通过分析内存分配、虚拟内存技术以及内存保护和分页机制等方面,揭示内存管理对提升系统性能的重要性,并提供了一系列优化内存使用效率的方法。
|
24天前
|
存储 算法 开发者
深入理解操作系统的内存管理机制
【4月更文挑战第3天】 本文旨在探讨操作系统中至关重要的一环——内存管理。不同于常规的技术分析文章,我们将从宏观和微观两个维度来剖析内存管理的核心原理及其对系统性能的影响。通过深入研究分页、分段以及虚拟内存等关键技术,我们揭示了操作系统如何优化资源分配,实现多任务并发执行的同时保证系统的稳定与高效。本文不仅适用于计算机科学专业的学者和学生,同时也为软件开发者提供了宝贵的参考,帮助他们设计出更高效的程序。
|
28天前
|
存储 算法
数据结构之动态内存管理机制(下)
数据结构之动态内存管理机制
17 1