在上篇我们完成了子弹和地图碰撞的检测,在这篇我们将完成敌方坦克AI的编写。

具体思路是屏幕中保持有四个敌方坦克,然后坦克随机方向运动,并且子弹消失后1秒发射一次

1.我们新建一个敌方坦克的AI类来控制地方坦克AI行为:

static const char* enemyTankType[] =
{
"normalU.png", "speedU.png", "armor1U.png"
}; class EnemyAI
{
public:
EnemyAI();
~EnemyAI(); static EnemyAI* createEnemyAIWithTank(Tank* tank);
void initEnemyAIWithTank(Tank* tank);
void update(float delta); private:
void addTank(float delta);
void tankAction(float delta); private:
CCArray* mEnemyTanks;
Tank* mTank;
TileMapInfo* mTileMapInfo; //出现地点
CCPoint bornPoint[3];
};

上面目前只添加三种类型的敌方坦克,

在成员变量中,mEnemyTanks存放了所有的坦克,

mTank存放了玩家坦克,最后mTileMapInfo存放了地图信息。

bornPoint存放了敌方坦克出身的三个位置。

2.先实现void EnemyAI::initEnemyAIWithTank(Tank* tank)来初始化我们的EnemyAI类:

void EnemyAI::initEnemyAIWithTank(Tank* tank)
{
mTank = tank;
mTileMapInfo = tank->getTileMapInfo();
mEnemyTanks = CCArray::createWithCapacity(4);
mEnemyTanks->retain(); //初始化出现地点
CCSize mapSize = mTileMapInfo->getTileMap()->getContentSize();
CCSize tileSize = mTileMapInfo->getTileMap()->layerNamed("layer_0")->getMapTileSize();
bornPoint[0] = ccp(tileSize.width, mapSize.height - tileSize.height);
bornPoint[1] = ccp(mapSize.width / 2, mapSize.height - tileSize.height);
bornPoint[2] = ccp(mapSize.width - tileSize.width, mapSize.height - tileSize.height);
}

在上面我们初始化了敌方坦克出身的三个地点,然后创建了一个容量为4的数组mEnemyTanks来存放敌方坦克对象。

3.再实现静态方法返回EnmeyAI类指针:

EnemyAI* EnemyAI::createEnemyAIWithTank(Tank* tank)
{
EnemyAI* enemyTank = new EnemyAI();
enemyTank->initEnemyAIWithTank(tank); return enemyTank;
}

4.实现void EnemyAI::addTank(float delta):

其中delta是每一帧调用之间的事件间隔

void EnemyAI::addTank(float delta)
{
static float deltaTimes = 0.0f;
deltaTimes += delta;
if (deltaTimes >= 2.0f)
{
deltaTimes = 0.0f;
int count = mEnemyTanks->count();
if (count < 3) //先从固定位置添加三个坦克
{
Tank* enemyTank = Tank::createTankWithTankType(enemyTankType[count], mTileMapInfo);
enemyTank->setPosition(bornPoint[count]);
enemyTank->setRotation(180.0f);
mEnemyTanks->addObject(enemyTank);
}
else if (count == 3) //第四个坦克随机添加
{
int tankTypeIndex = (int)(CCRANDOM_0_1() * 4) % 3;
Tank* enemyTank = Tank::createTankWithTankType(enemyTankType[tankTypeIndex], mTileMapInfo);
enemyTank->setPosition(bornPoint[tankTypeIndex]);
enemyTank->setRotation(180.0f);
mEnemyTanks->addObject(enemyTank);
}
}
}

可以看到deltaTimes累加时间,超过2秒就开始添加坦克,初始三个坦克从固定位置添加,

然后第四个坦克随机位置添加。

5.实现void EnemyAI::tankAction(float delta),来控制坦克行为:

void EnemyAI::tankAction(float delta)
{
CCObject* pObj;
CCARRAY_FOREACH(mEnemyTanks, pObj)
{
Tank* tank = (Tank*)pObj; //坦克自动移动,碰到墙壁自动换方向
int Rotation = tank->getRotation();
if (!tank->command((enumOrder)(Rotation / 90 + 1)))
{
int n = (int)(CCRANDOM_0_1() * 5) % 5;
if (n != 0)
tank->command((enumOrder)n);
} //每隔一秒开一次火
tank->setBulletDelta(tank->getBulletDelta() + delta);
if (tank->getBulletDelta() > 0.5)
{
//开火后,如果子弹在飞行中,归零计时
if (tank->command(cmdFire))
{
tank->setBulletDelta(0.0);
}
}
}
}

从mEnemyTanks中遍历坦克对象,然后向固定方向移动,遇到墙壁则随机换方向移动。

在下面,子弹消失后,隔0.5秒再次发射子弹。

6.最后实现我们的void EnemyAI::update(float delta),它会在每一帧切换的时候被调用。

void EnemyAI::update(float delta)
{
//坦克不足4个,补充坦克
addTank(delta); //坦克行为控制
tankAction(delta);
}

可以看到,简单的调用添加坦克函数和控制坦克行为函数。

7.下面我们把敌人的AI类整合到CityScene场景中:

bool  CityScene::init()
{
CCLayer::init(); //初始化地图信息
char szRound[260];
sprintf(szRound, "Round%d.tmx", mRound);
TileMapInfo* tileMapInfo = TileMapInfo::createMapInfoWithFile(szRound);
CCTMXTiledMap* tmxTileMap = tileMapInfo->getTileMap();
this->addChild(tmxTileMap); //在已有的地图上,创建玩家坦克
mPlayerTank[0] = Tank::createTankWithTankType("player2U.png", tileMapInfo); //放到地图中初始化位置
CCSize tileSize = tmxTileMap->getTileSize();
CCSize mapSize = tmxTileMap->getContentSize();
mPlayerTank[0]->setPosition(ccp(mapSize.width / 2 - tileSize.width * 3, tileSize.height)); //添加虚拟手柄的显示
mLayerPanel = Panel::create();
addChild(mLayerPanel, 3); //创建敌人坦克的AI
mEnemyAI = EnemyAI::createEnemyAIWithTank(mPlayerTank[0]); return true;
}

//创建敌人坦克的AI
mEnemyAI = EnemyAI::createEnemyAIWithTank(mPlayerTank[0]);

可以看到最后面创建了我们的AI。

8.我们在场景的update中调用AI的update函数

void CityScene::update(float delta)
{
CCLayer::update(delta);
//将控制面板中的mLayerPanel获取的命令传给坦克
if (mPlayerTank[0] != NULL)
mPlayerTank[0]->command(mLayerPanel->getOrder()); //调用敌人AI的update
mEnemyAI->update(delta);
}

到这里基本的坦克AI构建完成,我们运行下看看效果:

完整代码下载地址:

http://download.csdn.net/detail/yincheng01/6764095

cocos2d-x游戏开发系列教程-坦克大战游戏之敌方坦克AI的编写的更多相关文章

  1. HTML5游戏开发系列教程6(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-6/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...

  2. HTML5游戏开发系列教程7(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-7/ 今天我们将完成我们第一个完整的游戏--打砖块.这次教程中,将 ...

  3. HTML5游戏开发系列教程5(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-5/ 最终我决定准备下一篇游戏开发系列的文章,我们将继续使用can ...

  4. HTML5游戏开发系列教程4(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-4/ 这篇文章是我们继续使用canvas来进行HTML5游戏开发系 ...

  5. cocos2d-x游戏开发系列教程-坦克大战游戏启动界面的编写

    用前面介绍的方法,创建一个cocos2d-x项目,可以看到新项目内容如下图:

  6. HTML5游戏开发系列教程8(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-8/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...

  7. HTML5游戏开发系列教程10(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-10/ 最后我们将继续使用canvas来进行HTML5游戏开发系列 ...

  8. HTML5游戏开发系列教程9(译)

    原文地址:http://www.script-tutorials.com/html5-game-development-lesson-9/ 今天我们将继续使用canvas来进行HTML5游戏开发系列的 ...

  9. cocos2d-x游戏开发系列教程-坦克大战游戏之子弹的碰撞检测处理

    在上篇我们加上了简单的坦克之间的碰撞检测,这篇我们继续加上子弹之间, 子弹与坦克之间的碰撞检测,对于上一篇碰撞处理不太完美的地方我们继续改进. 1.子弹之间的碰撞 //玩家子弹和敌方子弹之间的碰撞 C ...

  10. cocos2d-x游戏开发系列教程-坦克大战游戏之所有坦克之间的碰撞检测

    上篇我们完成了简单的AI编写,但是各个坦克移动时之间是可以重合的, 这节课我们来完成坦克之间的碰撞检测,还是在上篇的EnemyAI中完成. 1.我先现在坦克类Tank中添加两个成员变量: CC_SYN ...

随机推荐

  1. 【转】Configuring VM Acceleration on Linux

    Configuring VM Acceleration on Linux Linux-based systems support virtual machine acceleration throug ...

  2. [翻译]Orchard-修改首页布局

    前言 Orchard在你的站点应用的默认主题叫做”Theme Machine”.该主题包括CSS样式和一个布局框架. Orchard 允许你选择或排除显示在你站点每个页面的布局. 默认情况下下图中蓝色 ...

  3. linux命令 收集

    https://jaywcjlove.github.io/linux-command/ 源码:https://github.com/jaywcjlove/linux-command Linux思维导图 ...

  4. iOS内存管理 ARC与MRC

    想驾驭一门语言,首先要掌握它的内存管理特性.iOS开发经历了MRC到ARC的过程,下面就记录一下本人对iOS内存管理方面的一些理解. 说到iOS开发,肯定离不开objective-c语言(以下简称OC ...

  5. TortoiseSVN和Eclipse中subversion无法协同使用

    环境: Eclipse版本Luna, 第一次安装subversion插件时, 使用了http://download.eclipse.org/releases/luna中的Subversive, 版本为 ...

  6. python优秀库 - 使用xmltodict解析xml文档

    上次讲到如何使用BeautifulSoup解析XML文档,今天发现另外一个python库xmltodict(https://github.com/martinblech/xmltodict)也很简单. ...

  7. boost::bind的使用方法

    bind - boost 头文件: boost/bind.hpp bind 是一组重载的函数模板.用来向一个函数(或函数对象)绑定某些参数. bind的返回值是一个函数对象. 它的源文件太长了. 看不 ...

  8. 【工具篇】利用DBExportDoc V1.0 For MySQL自动生成数据库表结构文档

    对于DBA或开发来说,如何规范化你的数据库表结构文档是灰常之重要的一件事情.但是当你的库,你的表排山倒海滴多的时候,你就会很头疼了. 推荐一款工具DBExportDoc V1.0 For MySQL( ...

  9. Centos6.4下tar包安装最新版Mysql5.6

    1.下载 mysql:http://www.mysql.com/downloads/ (须要注冊ORACLE账号) 版本号:mysql-advanced-5.6.21-linux-glibc2.5-x ...

  10. 引入工程报包导入异常:import javax.servlet.annotation.WebFilter;

    引入工程报包导入异常:import javax.servlet.annotation.WebFilter; (2013-02-21 16:38:00)   分类: java 今天上午导入了一个项目,用 ...