cocos2d-x游戏开发系列教程-坦克大战游戏之敌方坦克AI的编写
在上篇我们完成了子弹和地图碰撞的检测,在这篇我们将完成敌方坦克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的编写的更多相关文章
- HTML5游戏开发系列教程6(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-6/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...
- HTML5游戏开发系列教程7(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-7/ 今天我们将完成我们第一个完整的游戏--打砖块.这次教程中,将 ...
- HTML5游戏开发系列教程5(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-5/ 最终我决定准备下一篇游戏开发系列的文章,我们将继续使用can ...
- HTML5游戏开发系列教程4(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-4/ 这篇文章是我们继续使用canvas来进行HTML5游戏开发系 ...
- cocos2d-x游戏开发系列教程-坦克大战游戏启动界面的编写
用前面介绍的方法,创建一个cocos2d-x项目,可以看到新项目内容如下图:
- HTML5游戏开发系列教程8(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-8/ 这是我们最新一篇HTML5游戏开发系列文章.我们将继续使用c ...
- HTML5游戏开发系列教程10(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-10/ 最后我们将继续使用canvas来进行HTML5游戏开发系列 ...
- HTML5游戏开发系列教程9(译)
原文地址:http://www.script-tutorials.com/html5-game-development-lesson-9/ 今天我们将继续使用canvas来进行HTML5游戏开发系列的 ...
- cocos2d-x游戏开发系列教程-坦克大战游戏之子弹的碰撞检测处理
在上篇我们加上了简单的坦克之间的碰撞检测,这篇我们继续加上子弹之间, 子弹与坦克之间的碰撞检测,对于上一篇碰撞处理不太完美的地方我们继续改进. 1.子弹之间的碰撞 //玩家子弹和敌方子弹之间的碰撞 C ...
- cocos2d-x游戏开发系列教程-坦克大战游戏之所有坦克之间的碰撞检测
上篇我们完成了简单的AI编写,但是各个坦克移动时之间是可以重合的, 这节课我们来完成坦克之间的碰撞检测,还是在上篇的EnemyAI中完成. 1.我先现在坦克类Tank中添加两个成员变量: CC_SYN ...
随机推荐
- 学习MVC遇到的问题
修改电脑上的DNS配置: Opendns 首选DNS服务器和备用DNS服务器分别设置为208.67.222.222和208.67.220.220 google的8.8.8.8 首选DNS服务器和备用D ...
- web测试 结果存储类型为“Database”,但尚未指定结果储存库连接字符串
vs2010 Ultimate版带有web测试功能,可以对网站的性能以及负载进行测试. 在进行负载测试时提示“异常 LoadTestConnectStringMissingException 1 Lo ...
- 2013 南京邀请赛 C count the carries
/** 大意: 给定区间(a,b), 将其转化为二进制 计算从a+(a+1)+(a+2)....+(a+b-1),一共有多少次进位 思路: 将(a,b)区间内的数,转化为二进制后,看其每一位一共有多少 ...
- 单色VGA显示verilogHDL通用代码
今天做VGA,真是拼凑了好久啊.唉,总算完成了. 本来想偷懒移植,最后还是自己写的代码. //2015/12/13 //designer : pengxiaoen //function : vga c ...
- HBase性能测试
hbase org.apache.hadoop.hbase.PerformanceEvaluationUsage: java org.apache.hadoop.hbase.PerformanceEv ...
- Linux学习:netstat命令
Netstat 命令用于显示各种网络相关信息,如网络连接,路由表,接口状态等.对于开发来说,很多时候用于查看端口占用情况. 执行netstat命令,其输出结果可以分成两部分: 1)一是“Active ...
- JSON XML IO数据操作
一.XML解析 通过继承org.xml.sax.helpers.DefaultHandler类,覆写characters(),startDocument(),startElement(),endEle ...
- C# 继承细节
假定没有为类定义任何显式的构造函数,这样编译器就会为所有的类提供默认的构造函数,在后台会进行许多操作,编译器可以很好地解决层次结构中的所有问题,每个类中的每个字段都会初始化为默认值.但在添加了一个我们 ...
- Python之路Day10
本节主要内容:memcache&redis.RabbitMQ.twisted框架 1. memcache&redis 1.1 memcache Memcached 是一个高性能的分布式 ...
- Python之路Day5
一.时间复杂度 (1)时间频度: 一个算法花费的时间与算法中语句的执行次数成正比例,哪个算法中语句执行次数多,它花费的时间就多.一个算法中的语句执行次数称为语句频度或时间频度,记为T(n). (2)时 ...