body, table{font-family: 微软雅黑; font-size: 10pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}

【嵌套类+静态对象】释放
#include <iostream>
#include<stdio.h>
using namespace std;
class Singleton
{
        class AutoRelease;         //前向声明
        private:
                static Singleton * _pInstance;
                static AutoRelease _autoRelease;     // 这里只是声明一个变量
                Singleton();
                ~Singleton();
                class AutoRelease
                {
                        public:
                                AutoRelease()
                                {
                                        cout<<"AutoRelease()"<<endl;
                                }
                                ~AutoRelease()
                                {
                                        cout<<"~AutoRelease()"<<endl;
                                        if(_pInstance!=NULL)
                                                delete _pInstance;
                                }
                };
        public:
                static Singleton * getInstance();
};
Singleton * Singleton:: _pInstance = getInstance();
//Singleton * Singleton:: _pInstance = Singleton::getInstance();
Singleton::AutoRelease Singleton:: _autoRelease;     //必须要初始化
//这里才是真正的定义一个变量,类似 int a。
//非线程安全
Singleton * Singleton::getInstance()
{
        cout<<"getInstance()"<<endl;
        if(_pInstance == NULL)
        {
                _pInstance = new Singleton;
        }
        return _pInstance;
}
Singleton::Singleton()
{
        cout<<"Singleton()"<<endl;
}
Singleton::~Singleton()
{
        cout<<"~Singleton"<<endl;
}
【嵌套类+静态对象】释放
//静态对象函数结束自动调用析构函数释放
int main()
{
        cout<<endl;
        cout<<"main()"<<endl;
        printf(" pInstance = %p\n",Singleton::getInstance());
        cout<<endl;
        Singleton * p1 = Singleton::getInstance();
        Singleton * p2 = Singleton::getInstance();
        printf("p1 = %p\n",p1);
        printf("p2 = %p\n",p2);
        cout<<endl;
        return 0;
}
// 单例类在程序结束时可以显示调用析构函数来释放静态成员变量。但是这样做容易出错,容易忘记。而却也很难保证在delete之后,没有代码再调用析构函数。
// 最好的方法就是让这个类自己知道在合适的时候把自己删除,程序结束系统会自动释放所有的全局变量,这也包括静态成员变量。可以利用这个特性就可以实现在单例类中定义一个今天成员变量,它的唯一工作就是在析构函数中释放单例类对象。
【pthread_once + atexit】释放  (和平台相关,用到线程)
#include <stdlib.h>
int atexit(void (*function)(void));       //当 main 函数退出时,被注册的函数会自动调用
#include <iostream>
#include<stdlib.h>
using namespace std;
void func()
{
        cout<<"func()"<<endl;
}
int main()
{
        cout<<"main()"<<endl;      
        atexit(func);
 //当main退出时,被注册的函数会自动调用
        return 0;
}

#include <pthread.h>
pthread_once_t once_control = PTHREAD_ONCE_INIT;
int pthread_once ( pthread_once_t  *once_control, void  (*init_routine) (void));       //多次调用,只有第一次调用会执行参数函数

#include <iostream>
#include<pthread.h>
#include<stdio.h>
#include<stdlib.h>
using namespace std;
class Singleton
{
        private:
                Singleton();
                ~Singleton();
                static Singleton * _pInstance;
                static pthread_once_t _once;
        public:
                static Singleton * getInstance();
                static void init();
                static void destroy();
};
Singleton * Singleton :: _pInstance = NULL;
pthread_once_t Singleton :: _once = PTHREAD_ONCE_INIT;
Singleton::Singleton()
{
        cout<<"Singleton()"<<endl;
}
Singleton::~Singleton()
{
        cout<<"~Singleton()"<<endl;
}
void Singleton :: destroy()
{
        cout<<"destroy()"<<endl;
        if(_pInstance != NULL)
                delete _pInstance;
}

//线程安全
Singleton * Singleton :: getInstance()
{
        cout<<"getInstance()"<<endl;
        pthread_once(&_once,Singleton::init);
        return _pInstance;;
}
void Singleton::init()
{
        cout<<"init()"<<endl;
        atexit(Singleton::destroy);
        if(_pInstance == NULL)
        {
                _pInstance = new Singleton;           //在类里面进行调用
        }
}

int main()
{
        Singleton * p1 = Singleton::getInstance();
        Singleton * p2 = Singleton::getInstance();
        Singleton * p3 = Singleton::getInstance();
        printf("p1 = %p\n",p1);
        printf("p2 = %p\n",p2);
        printf("p3 = %p\n",p3);
        return 0;
}

单例类singleton自动释放的更多相关文章

  1. [转]单例模式——C++实现自动释放单例类的实例

    [转]单例模式——C++实现自动释放单例类的实例 http://www.cnblogs.com/wxxweb/archive/2011/04/15/2017088.html http://blog.s ...

  2. Singleton单例类模式

    body, table{font-family: 微软雅黑; font-size: 10pt} table{border-collapse: collapse; border: solid gray; ...

  3. [Android面试题-7] 写出一个Java的Singleton类(即单例类)

    1.首先明确单例的概念和特点: a>单例类只能有一个实例 b>单例类必须自己创建一个自己的唯一实例 c>单例类必须为其他所有对象提供这个实例 2.单例具有几种模式,最简单的两种分别是 ...

  4. Unity Singleton 单例类(Unity3D开发之二十)

    猴子原创,欢迎转载.转载请注明: 转载自Cocos2Der-CSDN,谢谢! 原文地址: http://blog.csdn.net/cocos2der/article/details/47335197 ...

  5. Unity Singleton 单例类(Unity3D开发)

    一.添加单例模板类 using UnityEngine; public class Singleton<T> : MonoBehaviour where T : MonoBehaviour ...

  6. 创建类模式(五):单例(Singleton)

    定义 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式一般情况下通过使用private的构造函数确保了在一个应用中只产生一个实例,并且是自行实例化. 和静态变量的区别 虽然 ...

  7. 设计模式(java) 单例模式 单例类

    ·单例类 单实例类,就是这个类只能创建一个对象,保证了对象实例的唯一性. 1.单例模式( Singleton Pattern) 是一个比较简单的模式, 其定义如下:Ensure a class has ...

  8. (七)boost库之单例类

    (七)boost库之单例类 一.boost.serialzation的单件实现 单例模式是一种常用的软件设计模式.在它的核心结构中只包含一个被称为单例类的特殊类.通过单例模式可以保证系统中一个类只有一 ...

  9. 设计模式的征途—1.单例(Singleton)模式

    单例模式属于创建型模式的一种,创建型模式是一类最常用的设计模式,在软件开发中应用非常广泛.创建型模式将对象的创建和使用分离,在使用对象时无需关心对象的创建细节,从而降低系统的耦合度,让设计方案更易于修 ...

随机推荐

  1. C#调用小票打印机

    using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using Sy ...

  2. Win10应用《纸书科学计算器》更新啦!

    <纸书科学计算器>在2016年8月拿了计算机设计大赛国家一等奖,现在仍记得我在答辩时还给评委们普及了一波UWP平台的知识.受此鼓励,这款应用也不会停下更新的脚步^_^.最近从1.9小幅升级 ...

  3. Ubuntu16.04桌面版 连接到ftp服务器

    Ftp服务器在不同的网段,需要临时添加网段 不同网段临时添加方法: root@xzrs:/home/rxf# ip addr add 10.1.2.127/24 dev enp0s25 电脑左侧“连接 ...

  4. mongodb研究(mongodb 内存数据库)

    本日志大部分都不是原创的转载复制的会带链接保持版权 工作中使用mongodb已经好久了,讽刺的是到了最后快离职的时候才有时间好好研究下源码.   印象:mongodb是一个内存数据库,数据都是放到内存 ...

  5. 重写(overwrite)、重载(overload)和覆盖(override)三者之间的区别

    覆盖:子类继承了父类的同名无参函数.当子类从父类继承了一个无参函数,而又定义了一个同样的无参函数,则子类定义的方法覆盖父类的方法,称为覆盖. 重载:子类继承了父类的同名有参函数.当子类继承了父类的一个 ...

  6. T-shirt again

    T-shirt again 标签(空格分隔): 软工实践 第一次获得小黄裳是在大一下的C++课上,见T-shirt 0.0... 这次在软工课上能再次获得小黄裳,是我没有想到的,个人觉得里面有蛮多的运 ...

  7. 【前端】vue.js实现按钮的动态绑定

    vue.js实现按钮的动态绑定 实现效果: 实现代码以及注释: <!DOCTYPE html> <html> <head> <title>按钮绑定< ...

  8. linux 搭建Java环境

    一.下载jdk/jre文件 下载链接 二.安装Java环境 1.解压文件到     /usr/java    目录 # tar zxvf jre-8u60-linux-x64.gz 2.配置环境变量 ...

  9. JAVA基础补漏--文件读取

    public class Test2 { public static void main(String[] args) throws IOException { FileInputStream fis ...

  10. svn的分支

    svn的分支使用 新建一个项目的时候,选择建立自带trunk,branches和tags文件夹的. 其中trunk作为主开发. 有需要的时候,从trunk创建分支到对应的branches下面,新建分支 ...