Cocos2d-x优化中多线程并发訪问
源自于Objective-C的Ref对象,须要使用AutoreleasePool进行内存管理,AutoreleasePool是非线程安全的,全部不推荐在子多线程中调用Ref对象的retain()、 release()和autorelease()等函数。另外,OpenGL上下文对象也是不支持线程安全的。
可是有的时候我们须要异步载入一些资源,比如:载入图片纹理、声音的预处理和网络请求数据等。
假设是异步载入图片纹理我们能够使用第20.4.4一节介绍的内容。但声音的预处理和网络请求数据等就须要自己通过多线程技术实现了。
Cocos2d-x引擎也提供了多线程技术。Cocos2d-x 3.x之前是使用第三方的pthread技术。
Cocos2d-x 3.x之后使用C++11新规范中的std::thread多线程技术,std::thread使用起来比較简单。
1.std::thread多线程技术
std::thread是C++11 引入了一个新的线程库。它提供了线程管理相关函数。std::thread库中还提供了std::mutex(相互排斥量)。通过std::mutex能够实现线程同步。
启动一个新的线程非常easy。当我们创建一个 std::thread 对象时候,它便会自行启动。创建线程std::thread 对象时。能够提供该线程的回调函数。以下代码实现了创建线程和线程函数的回调:
#include <thread>
#include <iostream> void callfn(){ ①
std::cout << "Hello thread! " << std::endl;
}
int main(){
std::thread t1(callfn); ②
t1.join(); ③
return 0;
}
上述代码第②行是创建t1线程对象。它的參数是函数指针callfn,假设须要。我们还能够为回调函数提供參数。
代码第①行是回调函数的定义。第③行代码t1.join()是将子线程与主线程合并,这样的合并能够使子线程运行完毕后才干继续运行主线程,这是为了避免子线程还在运行,主线程已经运行结束而撤销。
创建线程还能够使用堆的方式分配内存,代码例如以下:
void callfn(){
std::cout << "Hello thread! " << std::endl;
}
int main(){
std::thread* t1 = new std::thread(callfn); ①
t1->join();
delete t1; ②
t1 = nullptr; ③
return 0;
}
上述代码第①行是通过堆方式分配内存,即通过new运算符创建动态线程对象。因此须要在使用完毕的情况下释放对象。我们在代码第②行使用delete t1语句释放,释放完毕还以通过代码第③行t1 = nullptr设置指针变量,这样能够防止“野指针”。
2.异步预处理声音
std::thread线程Cocos2d-x中有非常多现实的应用。异步预处理声音,异步载入一些资源资源文件。异步载入图片纹理Cocos2d-x为我们提供了API,可是它们异步载入须要我们自己实现。
以下我们介绍一下异步预处理声音。
我们在前面20.5一节介绍了声音预处理和清除,在那一节中预处理声音是同步的。它会导致阻塞主线程,使用户的感觉会“卡”了一下。假设这个“卡”比較长。我们解决主线程阻塞问题。改善用户体验,我们能够异步预处理声音。
我们在20.5一节的案例中採用std::thread线程异步预处理声音,我们能够在AppDelegate中进行异步载入,改动之后的AppDelegate.h代码例如以下:
#include "cocos2d.h"
#include "SimpleAudioEngine.h"
using namespace CocosDenshion;
class AppDelegate : private cocos2d::Application
{
private:
std::thread *_loadingAudioThread; ①
void loadingAudio(); ②
public:
AppDelegate();
virtual ~AppDelegate();
… …
};
我们在第①行声明了私有的std::thread线程指针变量_loadingAudioThread。第②代码是声明了私有的异步预处理声音函数loadingAudio()。
改动之后的AppDelegate.cpp代码例如以下:
include "AppDelegate.h"
#include "HelloWorldScene.h" USING_NS_CC; AppDelegate::AppDelegate()
{
_loadingAudioThread = new std::thread(&AppDelegate::loadingAudio,this); ①
} AppDelegate::~AppDelegate()
{
_loadingAudioThread->join(); ②
CC_SAFE_DELETE(_loadingAudioThread); ③
} bool AppDelegate::applicationDidFinishLaunching() {
… …
return true;
}
void AppDelegate::applicationDidEnterBackground() {
Director::getInstance()->stopAnimation();
SimpleAudioEngine::getInstance()->pauseBackgroundMusic();
}
void AppDelegate::applicationWillEnterForeground() {
Director::getInstance()->startAnimation();
SimpleAudioEngine::getInstance()->resumeBackgroundMusic();
} void AppDelegate::loadingAudio() ④
{
//初始化 音乐
SimpleAudioEngine::getInstance()->preloadBackgroundMusic("sound/Jazz.mp3");
SimpleAudioEngine::getInstance()->preloadBackgroundMusic("sound/Synth.mp3");
//初始化 音效
SimpleAudioEngine::getInstance()->preloadEffect("sound/Blip.wav");
}
上述代码第①行是在构造函数里创建线程对象。创建线程对象代码也能够放置到 AppDelegate::applicationDidFinishLaunching()函数中,我们依据须要在合适的地方创建。
第②行代码_loadingAudioThread->join()是合并线程到主线程,这个处理是在析构函数中调用的,join()函数通常是在线程处理完毕后调用,我们能够在析构函数中调用。也能够在一些退出函数(如Layer的onExit函数)中调用。因为是_loadingAudioThread动态对象指针类型,须要释放对象,我们能够通过第③行代码CC_SAFE_DELETE(_loadingAudioThread)释放。CC_SAFE_DELETE宏的作用例如以下:
delete _loadingAudioThread;
_loadingAudioThread = nullptr;
第④行代码AppDelegate::loadingAudio() 定义了线程回调函数。我们在这个函数中预处理声音。
本书交流讨论站点:http://www.cocoagame.net
《Cocos2d-x实战 C++卷》现已上线。各大商店均已开售:
京东:http://item.jd.com/11584534.html
当当:http://product.dangdang.com/23606265.html
互动出版网:http://product.china-pub.com/3770734
《Cocos2d-x实战 C++卷》源代码及样章下载地址:
样章下载地址:http://51work6.com/forum.php?
mod=viewthread&tid=1157&extra=page%3D1
Cocos2d-x优化中多线程并发訪问的更多相关文章
- Cocos2d-x优化中多线程并发访问
多线程并发访问在Cocos2d-x引擎中用的不是很多,这主要是因为中整个结构设计没有采用多线程.源自于Objective-C的Ref对象,需要使用AutoreleasePool进行内存管理,Autor ...
- 利用httpclient和多线程刷訪问量代码
缘起于玩唱吧,由于唱吧好友少,訪问量低,又不想加什么亲友团之类的,主要是太麻烦了,于是我就琢磨唱吧的訪问机制,准备用java的httpclient库来进行刷訪问量,想到动态IP反复使用就想到了用多线程 ...
- nodejs中使用monk訪问mongodb
mongodb 安装mongodb 我认为还是用mannual install靠谱一点儿:http://docs.mongodb.org/manual/tutorial/install-mongodb ...
- 让devstack中的vm訪问外网
devstack默认会建立一个Public网络,地址为172.24.4.0/24,可是这个网络并非运营商分配给我们的网络.所以仅仅能通过nat的方式让devstack建立的虚拟机訪问外网. br-ex ...
- Nginx并发訪问优化
Nginx反向代理并发能力的强弱,直接影响到系统的稳定性.安装Nginx过程,默认配置并不涉及到过多的并发參数,作为产品执行,不得不考虑这些因素.Nginx作为产品执行,官方建议部署到Linux64位 ...
- Java中多线程并发体系知识点汇总
一.多线程 1.操作系统有两个容易混淆的概念,进程和线程. 进程:一个计算机程序的运行实例,包含了需要执行的指令:有自己的独立地址空间,包含程序内容和数据:不同进程的地址空间是互相隔离的:进程拥有各种 ...
- 在IntelliJ IDEA中多线程并发代码的调试方法
通常来说,多线程的并发及条件断点的debug是很难完成的,或许本篇文章会给你提供一个友好的调试方法.让你在多线程开发过程中的调试更加的有的放矢. 我们将通过一个例子来学习.在这里,我编写了一个多线程程 ...
- Jmeter 中多线程并发和循环控制器
今天遇到一个场景, 给一个手机号发送短信验证码,通过正确输入短信验证码即登录并获得token,进行其他操作. 短信验证码是4位,即9999个组合, 接口没有对验证次数做校验,所以可以一直一直尝试通过验 ...
- SharePoint 2010 在同意匿名訪问的站点中隐藏登陆链接
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/u012025054/article/details/37565787 SharePoint 2010 ...
随机推荐
- CentOS7安全设置 yum-cron系统自动更新,firewalld防火墙简单使用
PermitRootLogin nosystemctl restart sshd.service; yum -y install firewalld; systemctl start firewall ...
- Windows SVN变化邮件通知(Python2.7实现)
1,新增文件post-commit.bat 内容: rem REPOS-PATH (the path to this repository) set REPOS=%1 rem REV (the num ...
- ES6的模块化
在之前的 javascript 中一直是没有模块系统的,前辈们为了解决这些问题,提出了各种规范, 最主要的有CommonJS和AMD两种.前者用于服务器,后者用于浏览器.而 ES6 中提供了简单的模块 ...
- 【巧妙预处理系列】【UVA1330】City game
最大子矩阵(City Game, SEERC 2004, LA 3029) 给定一个m×n的矩阵,其中一些格子是空地(F),其他是障碍(R).找出一个全部由F组成的面积最大的子矩阵,输出其面积乘以3后 ...
- JVM 垃圾回收机制( 一) 回收对象的判定
关于JVM 的垃圾回收机制,我们一般都没过多深入,因为JAVA 和 C++ 的一个很大区别就是,JAVA 帮我们做了垃圾回收,而不用像C++ 那么样手动进行回收,当然任何自动的东西都存在一定弊端,比如 ...
- spring入门:beans.xml不提示、别名、创建对象的三种方式
spring的版本是2.5 一.beans.xml文件不提示 Location:spring-framework-2.5.6.SEC01\dist\resources\spring-beans-2.5 ...
- 改变页面选择文字颜色和背景颜色----selection伪元素
div::selection{color:#fff;background: #E83E84;text-shadow:none}
- SCII码表 键盘常用ASCII码
ASCII码对照表 在Web开发时,如下的ASCII码只要加上&#和;就可以变成Web可以辨认的字符了在处理特殊字符的时候特别有用,如:' 单引号在数据库查询的时候是杀手,但是如果转换成' ...
- C# 和Java的foreach的不同用法
循环语句为苦逼的程序猿们提供了很大的便利,有while.do...while.for和 foreach.而且foreach语句很简洁,但是它的优点不仅仅在于此,它的效率也是最高的. 作为两个开发语言, ...
- intellj idea maven 无效的目标发行版: 1.8
File ->settings->maven->runner->jre -> jdk1.8