对象管理类的原理是这种:

ObjectManager类是一个单例类,全局仅仅有一个对象实例存在。初始化的时候创建两个数组CCArray来保存金币和岩石。为什么要保存,由于在地图重载的时候。要销毁看不见的那些对象。金币和岩石是随机加入的,每一个金币和岩石都有一个地图索引,就是说它是在第几个地图上的,删除的时候依据这个来删除。

我的博客:http://blog.csdn.net/dawn_moon,欢迎转载

CCArray有个遍历的宏CCARRAY_FOREACH。假设你在遍历的时候进行增删的操作就会导致错误,这里我用了一个暂时的CCArray来保存要删除的对象。然后再遍历这个暂时数组对源数组进行删除,最后清空暂时数组。

  1. CCArray* tempCoins = CCArray::create();
  2. CCArray* tempRocks = CCArray::create();
  3. CCObject* itor;
  4. // 遍历CCArray的时候不能做增删操作,否则会出错
  5. // 这里分两步来完毕,第一次遍历,将要删除的对象存到一个暂时CCArray里
  6. // 第二步遍历暂时CCArray,将里面的对象从源数组里删掉,最后清空暂时数组
  7. CCARRAY_FOREACH(mCoins,itor)
  8. {
  9. Coin* coin = dynamic_cast<Coin*>(itor);
  10. if ((coin != NULL) && (coin->getMap() == mapIndex)) {
  11. tempCoins->addObject(coin);
  12. coin->destroy();
  13. }
  14. }
  15. CCARRAY_FOREACH(tempCoins,itor)
  16. {
  17. mCoins->fastRemoveObject(itor);
  18. }
  19. tempCoins->removeAllObjects();

这个类在PlayScene的init里面初始化:

  1. ObjectManager* ObjM= ObjectManager::sharedObjectManager();
  2. ObjM->initManager(spriteBatch, mWorld);
  3. ObjM->setObjectToMap(1, mMapManager->getMapWidth());

然后在PlayScene的update里面加入金币和岩石。

  1. // 假设发生地图重载。就回收废的金币和岩石,加入新的金币和岩石
  2. if (mMapManager->chechReloadMap(mLastEyeX)) {
  3. ObjectManager::sharedObjectManager()->recycleObjectOfMap(mMapManager->getCurMapIndex() - 1);
  4. ObjectManager::sharedObjectManager()->setObjectToMap(mMapManager->getCurMapIndex() + 1, mMapManager->getMapWidth());
  5. }

好了。看碰撞检測。box2d的碰撞检測由物理世界维护,物理世界能知道全部发生的碰撞事件,并用一个回调来处理。我们要自己处理碰撞检測。就要实现这个回调b2ContactListener

让PlayScene继承这个类。实现里面的一个函数BeginContact。然后给world设置碰撞监听器

mWorld->SetContactListener(this);

看下碰撞检測的实现:

  1. void PlayScene::BeginContact(b2Contact *contact)
  2. {
  3. //    CCLog("begin contact!");
  4. void* bodyUserDataA = contact->GetFixtureA()->GetBody()->GetUserData();
  5. void* bodyUserDataB = contact->GetFixtureB()->GetBody()->GetUserData();
  6. if (bodyUserDataA && bodyUserDataB)
  7. {
  8. B2Sprite* contactA = static_cast<B2Sprite*>(bodyUserDataA);
  9. BaseObject* obj = NULL;
  10. if (contactA == mRunner->getRunnerSprite())
  11. {
  12. obj = static_cast<BaseObject*>(bodyUserDataB);
  13. }else
  14. {
  15. obj = static_cast<BaseObject*>(bodyUserDataA);
  16. }
  17. if (COINTAG == obj->getObjSprite()->getTag()) {
  18. ((Status*)(this->getParent()->getChildByTag(STATUSTAG)))->addCoin(1);
  19. mRemoveObjs->addObject(obj);
  20. SimpleAudioEngine::sharedEngine()->playEffect(pickUpCoins);
  21. }else if(ROCKTAG == obj->getObjSprite()->getTag())
  22. {
  23. mRunner->die();
  24. unscheduleUpdate();
  25. mState = GameOverState;
  26. GameOver* over = GameOver::create();
  27. this->getParent()->addChild(over);
  28. }
  29. }
  30. }

碰撞发生后,通过body的用户数据来进行辨别。这个用户数据userData是一个void*指针。存放用户的不论什么数据。在做物理精灵的时候对它进行设置,就是为了这个时候用的。

地板的userData是NULL,所以假设这个userData不为NULL的话。那么它要么是Runner要么是金币要么是岩石。

Runner的userData是一个B2Sprite。金币和岩石的userData都是BaseObject的子类,所以做类型转换。再用tag来区分金币和岩石。假设是金币,就加入到一个CCArray里面。然后在update里面做清除,假设是岩石,就GameOver了。

分数和跑酷的距离是一个单独的CCLayer。由于PlayScene是一个无限的Layer。所以分数要单独出来,不然它会跑出屏幕。GameOver是一个CCLayerColor,由于要做一个有透明度的层。表示游戏结束了。

图我就不上了。稍后放出源代码。

cocos2d-x游戏开发 跑酷(八) 对象管理 碰撞检測的更多相关文章

  1. 基于cocos2d-x的跑酷游戏,不同高度地面的碰撞检測demo,有兴趣能够看一看

    1. demo大致分为4个模块: 地图,角色,障碍 逻辑检測认为和不同高度地面的碰撞.1次跳和2连跳的实现. 代码链接:http://download.csdn.net/detail/zangleng ...

  2. iOS cocos2d 2游戏开发实战(第3版)书评

    2013是游戏爆发的一年,手游用户也是飞速暴增.虽然自己不做游戏,但也是时刻了解手机应用开发的新动向.看到CSDN的"写书评得技术图书赢下载分"活动,就申请了一本<iOS c ...

  3. cocos2d-x ios游戏开发初认识(八) 触摸事件与碰撞检測

    玩过植物大战僵尸都知道,要在草坪里放一朵向日葵或者其他的植物仅仅需触摸那个植物将其拖入到想要摆放的位置,这事实上就是这节要写的触摸事件.还能够发现当我们的僵尸出来的时候,我们的小豌豆会发子弹攻击僵尸, ...

  4. cocos2d-x 3.0游戏实例学习笔记 《跑酷》第七步--物理碰撞检測(1)

    说明:这里是借鉴:晓风残月前辈的博客,他是将泰然网的跑酷教程,用cocos2d-x 2.X 版本号重写的,眼下我正在学习cocos2d-X3.0 于是就用cocos2d-X 3.0重写,并做相关笔记 ...

  5. Cocos2d-x3.0游戏实例之《别救我》第七篇——物理世界的碰撞检測

    事实上我也非常吃惊-居然写到第七篇了,我估计也就是四篇的内容,感觉非常奇妙,我也不会非常唠叨什么吖);    // 0001 );   // 0001 ); // 0001 这样我们才干监听到它们的碰 ...

  6. 8、Cocos2dx 3.0三,找一个小游戏开发3.0存储器管理的版本号

    重开发人员的劳动成果,转载的时候请务必注明出处:http://blog.csdn.net/haomengzhu/article/details/27693365 复杂的内存管理 移动设备上的硬件资源十 ...

  7. 游戏开发设计模式之对象池模式(unity3d 示例实现)

    前篇:游戏开发设计模式之命令模式(unity3d 示例实现) 博主才学尚浅,难免会有错误,尤其是设计模式这种极富禅意且需要大量经验的东西,如果哪里书写错误或有遗漏,还请各位前辈指正. 原理:从一个固定 ...

  8. 【Unity3D游戏开发】之全局管理类的几种方式 (十六)

    如何在Unity中实现全局管理类?由于Unity脚本的运行机制和面向组件编程(COP)的思想,实现起来和普通的方式略有差别. 第一种方式是使用静态类.适合存储一些全局的变量,如游戏当前关卡.玩家得分等 ...

  9. Unity游戏开发中的内存管理_资料

    内存是手游的硬伤——Unity游戏Mono内存管理及泄漏http://wetest.qq.com/lab/view/135.html 深入浅出再谈Unity内存泄漏http://wetest.qq.c ...

随机推荐

  1. HDU 1231.最大连续子序列-dp+位置标记

    最大连续子序列 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Sub ...

  2. scrapy抓取拉勾网职位信息(七)——数据存储(MongoDB,Mysql,本地CSV)

    上一篇完成了随机UA和随机代理的设置,让爬虫能更稳定的运行,本篇将爬取好的数据进行存储,包括本地文件,关系型数据库(以Mysql为例),非关系型数据库(以MongoDB为例). 实际上我们在编写爬虫r ...

  3. async await 使用笔记

    JavaScript的网络请求异步的,即网络请求不会阻塞当前 js 代码的继续执行,而是通过回调的方式,网络请求的代码块中注入回调函数,当网络请求完成,会触发相应的事件,通过触发事件来执行注册的回调函 ...

  4. Highcharts实现走势图

    Highcharts 是一个用纯JavaScript编写的一个图表库, 能够很简单便捷的在web网站或是web应用程序添加有交互性的图表,并且免费提供给个人学习.个人网站和非商业用途使用.HighCh ...

  5. 【BZOJ 3997】 3997: [TJOI2015]组合数学 (DP| 最小链覆盖=最大点独立集)

    3997: [TJOI2015]组合数学 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 919  Solved: 664 Description 给出 ...

  6. BZOJ2938 POI2000病毒

    我们不能让重复过的字串出现在无限串上(就叫这个了...) 也就是要自动机一直能匹配但就是匹配不到,那么就是在自动机上找一个环. dfs判环即可.注意是个有向图. #include<bits/st ...

  7. Codeforces 835 F. Roads in the Kingdom

    \(>Codeforces\space835 F. Roads in the Kingdom<\) 题目大意 : 给你一棵 \(n\) 个点构成的树基环树,你需要删掉一条环边,使其变成一颗 ...

  8. noip模拟 五子棋

    递推+模拟.在读取数据时,我们建4个图,分别代表这个图中横.纵.左斜右斜的连续长度.例如heng[i][j]代表ij这个点所在的横着一条线的长度. 然后搜索,对于一个空点,如果他的上下都>=4那 ...

  9. [转]android.support.v4.app.Fragment和android.app.Fragment区别

      1.最低支持版本不同 android.app.Fragment 兼容的最低版本是android:minSdkVersion="11" 即3.0版 android.support ...

  10. SVN 服务器搭建及使用 一

    SVN服务器搭建和使用(一) Subversion是优秀的版本控制工具,其具体的的优点和详细介绍,这里就不再多说. 首先来下载和搭建SVN服务器. 现在Subversion已经迁移到apache网站上 ...