C++ 的new 运算子和C 的malloc 函数都是为了配置内存,但前者比之后者的优点
是,new 不但配置对象所需的内存空间时,同时会引发构造式的执行。

所谓构造式(constructor),就是对象诞生后第一个执行(并且是自动执行)的函数,它
的函数名称必定要与类别名称相同。

相对于构造式,自然就有个析构式(destructor),也就是在对象行将毁灭但未毁灭之前
一刻,最后执行(并且是自动执行)的函数,它的函数名称必定要与类别名称相同,再
在最前面加一个~ 符号。

#include <iostream>
using namespace std;
class CDemo{
char name[];
public:
CDemo(){
cout<<"class begin !!!"<<endl;
}
CDemo(char * ch)
{
strcpy(name,ch);
cout << "Constructor called for " << name << '/n';
}
~CDemo(){
cout<<"desturctor called for "<<name<<endl;
}
};
void func()
{
CDemo LocalObjectInFunc("LocalObjectInFunc"); // in stack
static CDemo StaticObject("StaticObject"); // local static
CDemo* pHeapObjectInFunc = new CDemo("HeapObjectInFunc"); // in heap
cout << "Inside func" << endl;
}
CDemo GlobalObject("GlobalObject"); // global static
int main()
{
CDemo LocalObjectInMain("LocalObjectInMain"); // in stack
CDemo* pHeapObjectInMain = new CDemo("HeapObjectInMain"); // in heap
cout << "In main, before calling func/n";
func();
cout << "In main, after calling func/n";
system("pause");
}

以上代码的执行结果是:

Constructor called for GlobalObject
Constructor called for LocalObjectInMain
Constructor called for HeapObjectInMain
In main, before calling func
Constructor called for LocalObjectInFunc
Constructor called for StaticObject
Constructor called for HeapObjectInFunc
Inside func
desturctor called for LocalObjectInFunc
In main, after calling func
请按任意键继续. . .

所以可以得到以下几个结论:

  • 对于全域对象(如本例之GlobalObject),程序一开始,其构造式就先被执行
    (比程序进入点更早);程序即将结束前其析构式被执行。MFC 程序就有这
    样一个全域对象,通常以application object 称呼。
  • 对于区域对象,当对象诞生时,其构造式被执行;当程序流程将离开该对象的
    存活范围(以至于对象将毁灭),其析构式被执行。
  • 对于静态(static)对象,当对象诞生时其构造式被执行;当程序将结束时(此
    对象因而将遭致毁灭)其析构式才被执行,但比全域对象的析构式早一步执
    行。
  • 对于以new 方式产生出来的区域对象,当对象诞生时其构造式被执行。析构
    式则在对象被delete 时执行(上例程序未示范)。

四种不同的对象生存方式(in stack、in heap、global、local static)

在C++ 中,有四种方法可以产生一个对象。

第一种方法是在堆栈(stack)之中产生它:

void MyFunc()
   {
           CFoo foo; // 在堆栈(stack)中产生foo 对象
           ...
    }
第二种方法是在堆积(heap)之中产生它:

void MyFunc()
  {
       ...
      CFoo* pFoo = new CFoo(); // 在堆(heap)中产生对象
   }
第三种方法是产生一个全域对象(同时也必然是个静态对象):
        CFoo foo; // 在任何函数范围之外做此动作

第四种方法是产生一个区域静态对象:
   void MyFunc()
  {
       static CFoo foo; // 在函数范围(scope)之内的一个静态对象
       ...
  }

不论任何一种作法,C++ 都会产生一个针对CFoo 构造式的调用动作。前两种情况,C++
在配置内存-- 来自堆栈(stack)或堆积(heap)-- 之后立刻产生一个隐藏的(你的原
代码中看不出来的)构造式调用。第三种情况,由于对象实现于任何「函数活动范围
(function scope)」之外,显然没有地方来安置这样一个构造式调用动作。

是的,第三种情况(静态全域对象)的构造式调用动作必须靠startup 码帮忙。startup 码
是什么?是更早于程序进入点(main 或WinMain)执行起来的码,由C++ 编译器提供,
被联结到你的程序中。startup 码可能做些像函数库初始化、进程信息设立、I/O stream 产
生等等动作,以及对static 对象的初始化动作(也就是调用其构造式)。

当编译器编译你的程序,发现一个静态对象,它会把这个对象加到一个串行之中。更精

确地说则是,编译器不只是加上此静态对象,它还加上一个指针,指向对象之构造式及
其参数(如果有的话)。把控制权交给程序进入点(main 或WinMain)之前,startup 码
会快速在该串行上移动,调用所有登记有案的构造式并使用登记有案的参数,于是就初始化了你的静态对象。

第四种情况(区域静态对象)相当类似C 语言中的静态区域变量,只会有一个实体
(instance)产生,而且在固定的内存上(既不是stack 也不是heap)。它的构造式在
控制权第一次移转到其声明处(也就是在MyFunc 第一次被调用)时被调用。

原文地址:http://blog.csdn.net/j_factory/article/details/3582499

最新文章

  1. Asp.Net MVC及Web API框架配置会碰到的几个问题及解决方案
  2. SilverFoxServer出炉!!
  3. spring mvc 注解访问控制器以及接收form数据的方式,包括直接接收日期类型及对象的方法
  4. 自定义input[type=&quot;radio&quot;]的样式
  5. NOIP 2014 pj &amp; tg
  6. 微信支付开发(12) 认清微信支付v2和v3
  7. mysql in和or查询效率
  8. python方式实现scoket通信
  9. [AngularJS - app] AngularJS Location-picker app
  10. Ural 1046 Geometrical Dreams(解方程+计算几何)
  11. vue 2.0 无法编译ES6语法
  12. kafka使用实例
  13. 初涉IPC,了解AIDL的工作原理及使用方法
  14. Odoo 开源微信小程序商城模块
  15. Win7环境 搭建IIS环境。发布asp.net MVC项目到IIS(第一期)
  16. Python_守护进程、锁、信号量、事件、队列
  17. Python在金融量开源项目列表
  18. 【c++】删除string中指定的字符
  19. linux下安装Tomcat7.0
  20. hibernate执行createSQLQuery时字段重名的问题

热门文章

  1. c++ 的makefile文件实例
  2. SAP DBDI 网银接口实现案例
  3. GROUP BY、HAVING、AS 的用法小例子
  4. iPhone开发中,关于视图跳转的总结(转)
  5. 【转载】C#之玩转反射
  6. Cross-Site Scripting XSS 跨站攻击全攻略 分类: 系统架构 2015-07-08 12:25 21人阅读 评论(2) 收藏
  7. Erlang Trace机制
  8. u3d中 rect[2] == rt-&gt;GetGLWidth() &amp;&amp; rect[3] == rt-&gt;GetGLHeight()错误的原因及解决方法
  9. Service 保活法之二
  10. 非抢占式RCU中的一些概念