C++灵巧计数器技术

  • 时间:
  • 来源:互联网

阅读《large_scale_cpp_software_design》一书中灵巧计数器技术。该技术主要是针对非局部静态对象的静态对象成员的初始化。我们知道,非局部静态变量(即全局)是在程序启动前初始化,也就是在程序启动到进入main()之前,而局部静态对象,例如函数中的静态变量,对象的静态成员则是在使用时初始化。也就是说在程序启动时,非局部的静态对象初始化时并没有对其静态成员对象进行初始化。所以灵巧计数器的目的就是保证在非局部静态对象初始化之前,该对象的静态成员对象能被正确的初始化,然后程序结束时,静态成员对象能正确的被清理。

具体做法如下:
1.在非局部静态对象类的声明头文件中定义个静态的伪对象。

#ifndef __TEST__H__
#define __TEST__H__
#include <iostream>
using namespace std;


class PublicList
{
public:
    PublicList()
    {
        cout << "PublicList()" << endl;
    }

    ~PublicList()
    {
        cout << "~PublicList()" << endl;
    }
public:
    static int m_staticParam;
};


struct tagPublicInit
{
    tagPublicInit();
    ~tagPublicInit();
};

static tagPublicInit PublicInit;

#endif


实现如下:

#include "stdafx.h"
#include "Test.h"

static int nCount = 0;
int PublicList::m_staticParam = -1;

tagPublicInit::tagPublicInit()
{
    nCount++;
    cout << "tagPublicInit():nCount" << " " <<nCount << endl;

    PublicList::m_staticParam = 1;
    cout << "tagPublicInit():m_staticParam" << " " << PublicList::m_staticParam << endl;
}


tagPublicInit::~tagPublicInit()
{
    nCount--;
    cout << "~tagPublicInit():nCount" << " " << nCount << endl;

    if (nCount == 0)
    {
        PublicList::m_staticParam = 0;
        cout << "~tagPublicInit():m_staticParam" << " " << PublicList::m_staticParam << endl;
    }
}

所以当每次有cpp文件包含一次这个头文件时,在程序启动时,计数器nCount会加1.
然后当程序退出时。每个编译单元就会清理tagPublicInit对象,然后对计数器减一。当所有的包含该头文件的单元完成清理后,nCount变为0,然后对非局部的对象完成清理。

2.main.cpp

#include "stdafx.h"
#include "Test.h"

static PublicList gList;
int _tmain(int argc, _TCHAR* argv[])
{
    return 0;
}

3.程序运行结果
tagPublicInit():nCount 1
tagPublicInit():m_staticParam 1
tagPublicInit():nCount 2
tagPublicInit():m_staticParam 1
PublicList()
~PublicList()
~tagPublicInit():nCount 1
~tagPublicInit():nCount 0
~tagPublicInit():m_staticParam 0
请按任意键继续…

注:以上纯属个人理解,望指正。

本文链接http://element-ui.cn/news/show-576676.aspx