Cocos2d-x中的一个单例效果:

#ifndef __Moon3d__ParticleManager__

#define __Moon3d__ParticleManager__

#include "cocos2d.h"

USING_NS_CC;

class ParticleManager

{

public:

static ParticleManager* getInstance()//定义获取实例方法,单例设计模式.see notes

{

if ( m_pInstance == nullptr )//如果实例为空,

m_pInstance = new ParticleManager();//创建实例

return m_pInstance;//返回实例

}

private:

ParticleManager();//构造函数

static ParticleManager* m_pInstance;//粒子管理器实例

class CGarbo//内部类,主要作用是退出游戏的时候,清理内存,原理:程序在结束的时候,系统会自动析构所有的全局变量。事实上,系统也会析构所有的类的静态成员变量,就像这些静态成员也是全局变量一样。

{

public:

~CGarbo()//析构函数

{

if (ParticleManager::m_pInstance!= nullptr)//如果实例不为空

{

delete ParticleManager::m_pInstance;//清除单例

}

}

};

static CGarbo m_garbo;//定义内部类变量

public:

std::map<string,ValueMap> m_plistMap;//定义存放粒子数据的集合

void AddplistData(string strPlist,133); font-family:新宋体; font-size:9.5pt">string strName);//把粒子数据添加到集合里

ValueMap GetPlistData(string strName);//从粒子集合中获取粒子数据

};

#endif /* defined(__Moon3d__ParticleManager__) */

#include "ParticleManager.h"

ParticleManager* ParticleManager::m_pInstance=NULL;//变量初始化

ParticleManager::CGarbo ParticleManager::m_garbo;//变量初始化

ParticleManager()

{

m_plistMap.clear();//构造函数集合清理

}

void ParticleManager::string strName)

{

auto plistData=FileUtils::getInstance()->getValueMapFromFile(strPlist);//获取粒子数据

ValueMap>::iterator it = m_plistMap.begin();//获取集合

m_plistMap.insert(it,133); font-family:新宋体; font-size:9.5pt">pair<ValueMap>(strName,plistData));//把粒子数据存放到集合里

}

ValueMap string strplist)

{

auto plistData=m_plistMap.find(strplist)->second;//获取粒子数据

return plistData;//返回粒子数据

}

说明:

/*****************************************************************************

单例模式也称为单件模式、单子模式,可能是使用最广泛的设计模式。其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。有很多地方需要这样的功能模块,如系统的日志输出,GUI应用必须是单鼠标,MODEM的联接需要一条且只需要一条电话线,操作系统只能有一个窗口管理器,一台PC连一个键盘。

单例模式有许多种实现方法,在C++中,甚至可以直接用一个全局变量做到这一点,但这样的代码显的很不优雅。 使用全局对象能够保证方便地访问实例,但是不能保证只声明一个对象——也就是说除了一个全局实例外,仍然能创建相同类的本地实例。

《设计模式》一书中给出了一种很不错的实现,定义一个单例类,使用类的私有静态指针变量指向类的唯一实例,并用一个公有的静态方法获取该实例。

单例模式通过类本身来管理其唯一实例,这种特性提供了解决问题的方法。唯一的实例是类的一个普通对象,但设计这个类时,让它只能创建一个实例并提供对此实例的全局访问。唯一实例类Singleton在静态成员函数中隐藏创建实例的操作。习惯上把这个成员函数叫做Instance(),它的返回值是唯一实例的指针。

定义如下:

[cpp] view plaincopy

class CSingleton

{

private:

CSingleton() //构造函数是私有的

{

}

static CSingleton *m_pInstance;

public:

static CSingleton * GetInstance()

{

if(m_pInstance == NULL) //判断是否第一次调用

m_pInstance = new CSingleton();

return m_pInstance;

}

};

用户访问唯一实例的方法只有GetInstance()成员函数。如果不通过这个函数,任何创建实例的尝试都将失败,因为类的构造函数是私有的。GetInstance()使用懒惰初始化,也就是说它的返回值是当这个函数首次被访问时被创建的。这是一种防弹设计——所有GetInstance()之后的调用都返回相同实例的指针:

CSingleton* p1 = CSingleton :: GetInstance();

CSingleton* p2 = p1->GetInstance();

CSingleton & ref = * CSingleton :: GetInstance();

GetInstance稍加修改,这个设计模板便可以适用于可变多实例情况,如一个类允许最多五个实例。

单例类CSingleton有以下特征:

它有一个指向唯一实例的静态指针m_pInstance,并且是私有的;

它有一个公有的函数,可以获取这个唯一的实例,并且在需要的时候创建该实例;

它的构造函数是私有的,这样就不能从别处创建该类的实例。

大多数时候,这样的实现都不会出现问题。有经验的读者可能会问,m_pInstance指向的空间什么时候释放呢?更严重的问题是,该实例的析构函数什么时候执行?

如果在类的析构行为中有必须的操作,比如关闭文件,释放外部资源,那么上面的代码无法实现这个要求。我们需要一种方法,正常的删除该实例。

可以在程序结束时调用GetInstance(),并对返回的指针掉用delete操作。这样做可以实现功能,但不仅很丑陋,而且容易出错。因为这样的附加代码很容易被忘记,而且也很难保证在delete之后,没有代码再调用GetInstance函数。

一个妥善的方法是让这个类自己知道在合适的时候把自己删除,或者说把删除自己的操作挂在操作系统中的某个合适的点上,使其在恰当的时候被自动执行。

我们知道,程序在结束的时候,系统会自动析构所有的全局变量。事实上,系统也会析构所有的类的静态成员变量,就像这些静态成员也是全局变量一样。利用这个特征,我们可以在单例类中定义一个这样的静态成员变量,而它的唯一工作就是在析构函数中删除单例类的实例。如下面的代码中的CGarbo类(Garbo意为垃圾工人):

[cpp] view plaincopy

class CSingleton

{

private:

CSingleton()

{

}

static CSingleton *m_pInstance;

class CGarbo //它的唯一工作就是在析构函数中删除CSingleton的实例

{

public:

~CGarbo()

{

if(CSingleton::m_pInstance)

delete CSingleton::m_pInstance;

}

};

static CGarbo Garbo; //定义一个静态成员变量,程序结束时,系统会自动调用它的析构函数

public:

static CSingleton * GetInstance()

{

if(m_pInstance == NULL) //判断是否第一次调用

m_pInstance = new CSingleton();

return m_pInstance;

}

};

CGarbo被定义为CSingleton的私有内嵌类,以防该类被在其他地方滥用。

程序运行结束时,系统会调用CSingleton的静态成员Garbo的析构函数,该析构函数会删除单例的唯一实例。

使用这种方法释放单例对象有以下特征:

在单例类内部定义专有的嵌套类;

在单例类内定义私有的专门用于释放的静态成员;

利用程序在结束时析构全局变量的特性,选择最终的释放时机;

使用单例的代码不需要任何操作,不必关心对象的释放。

进一步的讨论

但是添加一个类的静态对象,总是让人不太满意,所以有人用如下方法来重新实现单例和解决它相应的问题,代码如下:

[cpp] view plaincopy

class CSingleton

{

private:

CSingleton() //构造函数是私有的

{

}

public:

static CSingleton & GetInstance()

{

static CSingleton instance; //局部静态变量

return instance;

}

};

使用局部静态变量,非常强大的方法,完全实现了单例的特性,而且代码量更少,也不用担心单例销毁的问题。

但使用此种方法也会出现问题,当如下方法使用单例时问题来了,

Singleton singleton = Singleton :: GetInstance();

这么做就出现了一个类拷贝的问题,这就违背了单例的特性。产生这个问题原因在于:编译器会为类生成一个默认的构造函数,来支持类的拷贝。

最后没有办法,我们要禁止类拷贝和类赋值,禁止程序员用这种方式来使用单例,当时领导的意思是GetInstance()函数返回一个指针而不是返回一个引用,函数的代码改为如下:

[cpp] view plaincopy

class CSingleton

{

private:

CSingleton() //构造函数是私有的

{

}

public:

static CSingleton * GetInstance()

{

static CSingleton instance; //局部静态变量

return &instance;

}

};

但我总觉的不好,为什么不让编译器不这么干呢。这时我才想起可以显示的声明类拷贝的构造函数,和重载 = 操作符,新的单例类如下:

[cpp] view plaincopy

class CSingleton

{

private:

CSingleton() //构造函数是私有的

{

}

CSingleton(const CSingleton &);

CSingleton & operator = (const CSingleton &);

public:

static CSingleton & GetInstance()

{

static CSingleton instance; //局部静态变量

return instance;

}

};

关于Singleton(const Singleton); Singleton & operate = (const Singleton&);函数,需要声明成私有的,并且只声明不实现。这样,如果用上面的方式来使用单例时,不管是在友元类中还是其他的,编译器都是报错。

不知道这样的单例类是否还会有问题,但在程序中这样子使用已经基本没有问题了。

考虑到线程安全、异常安全,可以做以下扩展

[cpp] view plaincopy

class Lock

{

private:

CCriticalSection m_cs;

public:

Lock(CCriticalSection cs) : m_cs(cs)

{

m_cs.Lock();

}

~Lock()

{

m_cs.Unlock();

}

};

class Singleton

{

private:

Singleton();

Singleton(const Singleton &);

Singleton& operator = (const Singleton &);

public:

static Singleton *Instantialize();

static Singleton *pInstance;

static CCriticalSection cs;

};

Singleton* Singleton::pInstance = 0;

Singleton* Singleton::Instantialize()

{

if(pInstance == NULL)

{ //double check

Lock lock(cs); //lock实现线程安全,用资源管理类,实现异常安全

//使用资源管理类,在抛出异常的时候,资源管理类对象会被析构,析构总是发生的无论是因为异常抛出还是语句块结束。

if(pInstance == NULL)

{

pInstance = new Singleton();

}

}

return pInstance;

}

之所以在Instantialize函数里面对pInstance 是否为空做了两次判断,因为该方法调用一次就产生了对象,pInstance == NULL 大部分情况下都为false,如果按照原来的方法,每次获取实例都需要加锁,效率太低。而改进的方法只需要在第一次 调用的时候加锁,可大大提高效率。

*/

1.Cocos2d-x-3.2编写3d打飞机,粒子管理器代码的更多相关文章

  1. 在ios上使用来自框架的boost :: filesysystem路径

    我一直在使用Boost作为PeteGoodliffe脚本构建的框架已有一段时间了.效果很好.最近我遇到了一个问题,可以通过将以下代码放入另一个全新的XCode项目中的视图控制器的viewDidLoad中来重现:当路径对象被销毁时会导致EXC_BAD_ACCESS.有没有其他人遇到这个问题?

  2. 如何在Xcode中追踪“libc abi.dylib:纯虚函数!”

    我有一个多线程OSX应用程序,它使用C,Objective-C和Swift的混合.当我的应用程序关闭时,我在Xcode调试器窗口中看到了这一点:我知道这个错误通常是由对C类构造函数或析构函数中的虚函数的调用引起的.有没有一种简单的方法可以找到它的位置?

  3. swift的struct结构体类型介绍使用

  4. Swift学习笔记(十三)析构过程

    析构过程在一个类的实例被释放之前,析构函数被立即调用。用关键字deinit来标示析构函数,类似于初始化函数用init来标示。析构过程原理Swift会自动释放不再需要的实例以释放资源Swift通过自动引用计数处理实例的内存管理示例析构函数是在实例释放发生前一步被自动调用。即使子类没有提供自己的析构函数,父类的析构函数也总是被调用。

  5. swift详解之七------------你真的了解构造过程吗构造过程和析构过程

    你真的了解构造过程吗注:本文为作者整理,尽量没有废话,都是干货。Swift提供了两种类型的类构造器来确保所有类实例中存储型属性都能获得初始值,它们分别是指定构造器和便利构造器。自动构造器的继承子类不会默认继承父类的构造器。

  6. 01-swift的简单了解

    swift是完全面向对象的语言,自身具有构造函数和析构函数。",表示不确定有没有对该变量进行赋值varstr:String?可选类型的使用在使用可选变量时,需要在变量名后面加上感叹号"!"成立时的代码":"不成立时的代码"区间运算符闭区间“...”包含最后一个值半闭区间"..

  7. Swift 2.0学习笔记Day 40——析构函数

    在析构过程中也会调用一种特殊的方法deinit,称为析构函数。下面看看示例代码:析构函数的调用是在实例被赋值为nil,表示实例需要释放内存,在释放之前先调用析构函数,然后再释放。运行结果如下:长方形:320.0x480.0调用析构函数...长方形:320.0x480.0调用析构函数...析构函数只适用于类,不能适用于枚举和结构体。而在Swift中,内存管理采用自动引用计数,不需要在析构函数释放不需要的实例内存资源,但是还是有一些清除工作需要在这里完成,如关闭文件等处理。

  8. 《从零开始学Swift》学习笔记Day 40――析构函数

    在析构过程中也会调用一种特殊的方法deinit,称为析构函数。下面看看示例代码:析构函数的调用是在实例被赋值为nil,表示实例需要释放内存,在释放之前先调用析构函数,然后再释放。运行结果如下:长方形:320.0x480.0调用析构函数...长方形:320.0x480.0调用析构函数...析构函数只适用于类,不能适用于枚举和结构体。而在Swift中,内存管理采用自动引用计数,不需要在析构函数释放不需要的实例内存资源,但是还是有一些清除工作需要在这里完成,如关闭文件等处理。

  9. Swift - deinit

    每个类只有一个析构函数,用关键字deinit定义。在类实例被释放之前被自动调用,不允许主动调用自己的析构函数。(即使子类没有提供自己的析构函数,父类的析构函数也总是被调用。

  10. Swift学习:2.15 析构过程

    用关键字deinit来标示析构函数,类似于初始化函数用init来标示。析构函数只适用于类类型。析构过程原理Swift会自动释放不再需要的实例以释放资源。如自动引用计数那一章描述,Swift通过自动引用计数处理实例的内存管理。vendCoins方法在bank分发硬币之前检查是否有足够的硬币。vendCoins方法返回一个整型值,表明了提供的硬币的实际数目。这里,player已经赢得了2,000硬币。在这发生前一步,其析构函数被自动调用,其硬币被返回到银行。

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部