在上篇我们完成了子弹和地图碰撞的检测,在这篇我们将完成敌方坦克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. string.Format()字符串格式化

    Format()基本语法:     {索引[,对齐][:格式字符串]}     ·索引:表示引用的对象列表中的第n个对象参数.     ·对齐(可选):设置宽度与对齐方式,该参数为带符号的整数.正数为 ...

  2. FreeCodeCamp:Confirm the Ending

    要求: 检查一个字符串(str)是否以指定的字符串(target)结尾. 如果是,返回true;如果不是,返回false. 结果: confirmEnding("Bastian", ...

  3. 五毛的cocos2d-x学习笔记05-场景与场景动画,动作

    场景切换函数: Director->getInstance()->replaceScene(Scene*); Director->getInstance()->runWithS ...

  4. The Power of Reading——英语学习小技巧之七

    This method is "The Power of Reading" and it comes from an article by Dr.Stephen Krashen. ...

  5. CoffeeScript 入门笔记

    写在前面: 被英文版指南坑了...闹了很久才明白.coffee怎么用.安装前需要有稳定版 Node.js, 和 npm (Node Package Manager). 借助 npm 可以安装 Coff ...

  6. python成长之路17

    一:web框架的本质,对于所有的Web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端. 1.1:python实现: #!/usr/bin/env python ...

  7. r语言之散点图类型type参数

    type参数用来控制所生成散点图的类型,有如下几个选项: type=“p”表示绘制单独的点 type=“l”表示绘制点连成的折线 type=“b”表示有线连接的点 type=“o”表示将点绘在线上 t ...

  8. IT第二十一天 - Collections、ArrayList集合、LinkedList集合、Set集合、HashMap集合、集合的操作注意【修20130828】

    NIIT第二十一天 上午 集合 1. 集合Collection存储数据的形式是单个存储的,而Map存储是按照键值对来存储的,键值对:即键+值同时存储的,类似align="center&quo ...

  9. 响应式Web图形篇 —— icon fonts 的探析及应用

    前言 像素完美(Pixel Perfection).分辨率无关(Resolution Independent)和多平台体验一致性是设计师们的追求. 可访问性(Accessability).加载性能和重 ...

  10. Lua,Lua API,配置文件

    想像一个场景:你的c程序须要有一个窗体,你想让用户能够自己定义窗体大小.方法非常多.比方使用环境变量,或键值对的文件. 无论如何,你须要解析它. 使用lua配置文件是个不错的选择. 首先,你能够定义例 ...