new对象数组时的内存布局

[cpp] view plain copy

    1. #include <iostream>
    2. #include <limits>
    3. using namespace std;
    4. #define SAFE_DELETE(x) \
    5. { \
    6. if (NULL != (x)) \
    7. { \
    8. delete (x); \
    9. (x) = NULL; \
    10. } \
    11. }
    12. #define SAFE_DELETE_ARY(x) \
    13. { \
    14. if (NULL != (x)) \
    15. { \
    16. delete[] (x); \
    17. (x) = NULL; \
    18. } \
    19. }
    20. class CA
    21. {
    22. public:
    23. CA()
    24. {
    25. setter_m_iTest(-1);
    26. }
    27. ~CA() {}
    28. void setter_m_iTest(int iIn) {m_iTest = iIn;}
    29. int getter_m_iTest() {return m_iTest;}
    30. private:
    31. int m_iTest;
    32. };
    33. void fnTestClass();
    34. void clear_cin();
    35. int main(int argc, char** argv, char** envp)
    36. {
    37. fnTestClass();
    38. cout << "END, press any key to quit" << endl;
    39. clear_cin();
    40. getchar();
    41. return 0;
    42. }
    43. void fnTestClass()
    44. {
    45. int i = 0;
    46. int iTmp = 0;
    47. const int iCntAry = 4;
    48. CA* pCaAry = NULL;
    49. CA* pCa = NULL;
    50. /** new 单个对象时的内存布局
    51. +   pCa 0x00883e30
    52. 00883E30  01 00 00 00  .... ///< pCa
    53. */
    54. pCa = new CA;
    55. if (NULL != pCa)
    56. {
    57. pCa->setter_m_iTest(1);
    58. SAFE_DELETE(pCa); ///< delete后, 原来的数据变为 EE FE EE FE
    59. }
    60. /** new对象数组时的内存布局
    61. +   pCaAry  0x007d3ddc
    62. 007D3DD8  04 00 00 00  .... ; ///< 类数组元素个数 = 4
    63. 007D3DDC  00 00 00 00  .... ; ///< 类数组开始, CA[0]
    64. 007D3DE0  01 00 00 00  .... ; ///< CA[1]
    65. 007D3DE4  02 00 00 00  .... ; ///< CA[2]
    66. 007D3DE8  03 00 00 00  .... ; ///< CA[3]
    67. */
    68. pCaAry = new CA[iCntAry];
    69. if (NULL != pCaAry)
    70. {
    71. for (i = 0; i < iCntAry; i++)
    72. {
    73. pCaAry[i].setter_m_iTest(i);
    74. }
    75. }
    76. SAFE_DELETE_ARY(pCaAry);
    77. /** 手工模拟delete删除单个对象
    78. */
    79. pCa = new CA;
    80. if (NULL != pCa)
    81. {
    82. pCa->setter_m_iTest(1);
    83. if (NULL != pCa)
    84. {
    85. pCa->~CA();
    86. free(pCa); ///< ok
    87. pCa = NULL;
    88. }
    89. }
    90. /** 手工模拟delete删除对象数组
    91. 会报错, 和实际情况不同
    92. 以后再研究 :(
    93. */
    94. // 会报错, 和实际情况不同
    95. //     pCaAry = new CA[iCntAry];
    96. //     if (NULL != pCaAry)
    97. //     {
    98. //         iTmp = *((int*)pCaAry - 1);
    99. //         for (i = 0; i < iTmp; i++)
    100. //         {
    101. //             pCaAry[i].setter_m_iTest(i);
    102. //             pCaAry[i].~CA();
    103. //         }
    104. //
    105. //         free(pCaAry); ///< ! crash
    106. //         pCaAry = NULL;
    107. //     }
    108. // 会报错, 和实际情况不同
    109. //     pCaAry = new CA[iCntAry];
    110. //     if (NULL != pCaAry)
    111. //     {
    112. //         iTmp = *((int*)pCaAry - 1);
    113. //         for (i = 0; i < iTmp; i++)
    114. //         {
    115. //             pCaAry[i].setter_m_iTest(i);
    116. //             pCaAry[i].~CA();
    117. //             free(pCaAry + i); ///< ! crash
    118. //         }
    119. //
    120. //         pCaAry = NULL;
    121. //     }
    122. }
    123. void clear_cin()
    124. {
    125. cin.clear();
    126. cin.sync();
    127. }

http://blog.csdn.net/lostspeed/article/details/50300867

时间: 2024-11-07 12:43:22

new对象数组时的内存布局的相关文章

【深入理解JVM】:Java对象的创建、内存布局、访问定位

对象的创建 一个简单的创建对象语句Clazz instance = new Clazz();包含的主要过程包括了类加载检查.对象分配内存.并发处理.内存空间初始化.对象设置.执行ini方法等. 主要流程如下: 1. 类加载检查 JVM遇到一条new指令时,首先检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载.解析和初始化过.如果没有,那必须先执行相应的类的加载过程. 2. 对象分配内存 对象所需内存的大小在类加载完成后便完全确定(对象内存布局),

浅谈对象的创建、内存布局和访问定位

在此简单的记录一下<深入理解Java虚拟机>第2章的2.3节内容. 对象的创建 这里的对象的创建是指普通的对象(不包括数组和Class对象).对象的创建简单来说就是执行new的时候,虚拟机做出对应的响应.让我们看看一下虚拟机创建对象的过程: 1.虚拟机遇到new指令时,首先尝试在常量池中定位到对应类的符号引用,并检查这个符号引用代表类是否已被加载.解析和初始化过.如果没有,那必须先执行相应的类加载过程(后续会写一下关于类加载的问题). 2.类加载检查通过后,为新生对象分配内存.对象内存的大小在

JVM 对象的创建、内存布局

一.对象的创建过程 一个简单的创建对象语句Clazz instance = new Clazz();包含的主要过程包括了类加载检查.对象分配内存.并发处理.内存空间初始化.对象设置.执行init方法等. 主要流程如下: 1. 类加载检查 JVM遇到一条new指令时,首先检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载.解析和初始化过.如果没有,那必须先执行相应的类的加载过程. 2. 对象分配内存 对象所需内存的大小在类加载完成后便完全确定(对象内

派生表中第一个基类没有虚函数,派生类存在虚函数时的内存布局

单继承的例子: #include <iostream> using namespace std; class A { public: A() { a = 1; ch = 'a'; //ASCII码97 } private: int a; char ch; }; class C : public A { public: C() { c = 3; } virtual void print() { cout << "C" << endl; } privat

深入理解Java虚拟机(二)、Java对象的创建,内存布局和访问定位

对象的创建: Object obj = new Object(); 常量池中是否有Ljava.lang.Object

C++对象数组与对象指针

(一)对象数组 将具有相同类类型的对象有序地集合在一起便构成了对象数组,以一维对象数组为例,其定义形式为: 类名 对象数组名[]; Point points[100]; 关于对象数组的几点说明: (1)在建立对象数组的时候需要调用构造函数.如果对象数组有100个元素,就需要调用100次构造函数. (2)如果对象数组所属类有带参数的构造函数时,可用初始化列表按顺序调用构造函数,使用复制初始化来初始化每个数组元素. Point A[3]={Point(0,0),Point(1,1),Point(2,

elf文件格式和运行时内存布局

elf的类型: Linux下的可执行文件格式叫做elf,全称是Executable Linkable Format.其实不仅仅是可执行文件,linux中常见的目标文件有目标文件.o.可执行文件如/bin/bash.共享目标文件(.so).和核心转储文件core dump.可以使用file命令确定文件格式. elf的结构: 如其结构如下图所示: ELF file format: +---------------+ | File header | +---------------+ | .text

虚继承之单继承的内存布局(VC在编译时会把vfptr放到类的头部,这和Delphi完全一致)

C++2.0以后全面支持虚函数与虚继承,这两个特性的引入为C++增强了不少功能,也引入了不少烦恼.虚函数与虚继承有哪些特性,今天就不记录了,如果能搞了解一下编译器是如何实现虚函数和虚继承,它们在类的内存空间中又是如何布局的,却可以对C++的了解深入不少.这段时间花了一些时间了解这些玩意,搞得偶都,不过总算有些收获,嘿嘿. 先看一段代码class A{      virtual aa(){};}; class B : public virtual  A{      char j[3];      

C++(二十六) — 构造函数、析构函数、对象数组

1.构造函数 (1)每个类都要定义它自己的构造函数和析构函数,是类的成员函数. 特点:名称与类名相同:没有返回值:一定是共有函数,可以直接访问类内所有成员函数:可以带默认形参,可以重载: class clock { public: // 无参构造函数 clock() { cout << "无参构造函数" << endl; } // 有参构造函数 clock(int h, int m , int s) { hour = h; minute = m; second