在上篇我们完成了子弹和地图碰撞的检测,在这篇我们将完成敌方坦克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. 【转】Ubuntu常用软件合集

    [转]Ubuntu常用软件合集 Ubuntu常用软件合集 我用的使Ubuntu-Kylin14.04,原因呢主要是觉得使本土化的,自带了日历.输入法.优客助手等易于上手的应用.也省的每次安装完原生的系 ...

  2. CSS样式表基础知识、样式表的分类及选择器

    一.CSS基本概念: CSS(Cascading Style Sheet,叠层样式表),作用是美化HTML网页. /*注释区域*/    此为注释语法 二.样式表的分类 1.内联样式表(和html联合 ...

  3. PHP开发中常见的安全问题详解和解决方法(如Sql注入、CSRF、Xss、CC等

    页面导航: 首页 → 网络编程 → PHP编程 → php技巧 → 正文内容 PHP安全 PHP开发中常见的安全问题详解和解决方法(如Sql注入.CSRF.Xss.CC等) 作者: 字体:[增加 减小 ...

  4. shell:监控进程运行状态并自动重启进程

    #!/bin/sh MAXRSTCOUNT=; PROCTOGO=/mnt/hgfs/code/test/show #count is the counter of test started time ...

  5. ubuntu之安装java浏览器插件

    最近搞什么openstack,在浏览器访问远程虚拟机的时候,需要浏览器有支持java.这个之前真没注意过呢, 通过自己的实践写点东西,方便一下你们搞: 1,首先去http://www.java.com ...

  6. [C#参考]字符编码

    微软的那个臭屁的JOEL(就是写<JOEL说软件>的那个牛人)曾说:“每一位软件开发人员必须.绝对要至少具备UNICODE与字符集知识(没有任何例外)”,我也常常困扰于字符集的转换等很多问 ...

  7. SQLSERVER 跨服务器查询

    SELECT * FROM OPENDATASOURCE(         'SQLOLEDB',         'Data Source=IP;User ID=UserId;Password=Pa ...

  8. 重新生成IE02

    procedure ReBuild_IE02( pi_aac001 in number, po_fhz out varchar2, po_msg out varchar2) is type typ_t ...

  9. QT实现窗口缩放打开与关闭(重叠窗口,太有意思了)

    基本思想:假设A为主窗口,B为子窗口.A打开或关闭时,先对A窗口进行截图,然后将图片部满整个B窗口的,在paintEvent里面进行动态缩放或放大画图.最后使用动画,将B窗口以动画的形式打开或关闭,动 ...

  10. perl 自动发产品

    use Net::SMTP; use LWP::UserAgent; use HTTP::Cookies; use HTTP::Headers; use HTTP::Response; use Enc ...