久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2154|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

淺談狀態(tài)機FSM設(shè)計方法

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:107189 發(fā)表于 2016-3-6 02:07 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式

狀態(tài)機,在游戲里面是非常重要的,最簡單的狀態(tài)機,莫過于

switch()
case 1:
    if(not 反復(fù)執(zhí)行狀態(tài)1)
       進入1狀態(tài)前要做的準備

    進入1狀態(tài)的過程

    if(not 反復(fù)執(zhí)行狀態(tài)1)
       離開狀態(tài)1的過程

  case2:
...

但這種方式不能很有效預(yù)定義所有的狀態(tài),也不能把這些狀態(tài)之間的切換過程合理的定義出來,“狀態(tài)”本身沒有一個合理的定義,幾乎是一種面向過程的方式,只過這種方式足夠簡單,也最容易讓人接受,缺點就沒有“狀態(tài)”的定義和指派功能,導(dǎo)致狀態(tài)的混亂,出現(xiàn)狀態(tài)處理重復(fù)代碼,甚至處理不一致的問題,按照OO的觀念,狀態(tài)描述本來就應(yīng)該是一種實體

比如“吃飯”這種狀態(tài),進入要做什么,進行時要做什么,退出時要做什么,需要進行一個描述,以下是我寫的狀態(tài)機管理策略:

//狀態(tài)的定義
class State
{
public:
State();
State(const char * name);
//狀態(tài)的名字
std::string statename;
//比較兩個狀態(tài)是否相同
inline bool operator ==(const State& other);
};


//狀態(tài)機基類
class BaseFsm
{
public:
//狀態(tài)機的狀態(tài)描述
State state;
//進入狀態(tài)
virtual void Enter(FsmEntity *entity);
//執(zhí)行狀態(tài)
virtual void Execute(FsmEntity *entity);
//離開狀態(tài)
virtual void Exit(FsmEntity *entity);
//比較兩個狀態(tài)機是否相同
inline bool operator ==(const BaseFsm& other);
};

//狀態(tài)機實體
class FsmEntity
{
protected:
  //當前所使用的狀態(tài)機
  BaseFsm *mCurrentFsm;
public:
  //構(gòu)造函數(shù)
  FsmEntity();
  //析構(gòu)函數(shù)
  virtual ~FsmEntity();
  //設(shè)置開始狀態(tài)
  void InitState(BaseFsm *fsm);
  //狀態(tài)是否初始化了
  bool IsStateInited();
  //保持狀態(tài)的方法
  void KeepState();
  //返回當前的狀態(tài)機
  BaseFsm * GetCurrentFsm();
  //改變狀態(tài)
  void ChangeState(BaseFsm *newFsm);
};

//狀態(tài)機容器
class FsmManager
{
private:
//狀態(tài)機容器的名稱
std::string name;
//所有的狀態(tài)集合
std::map<std::string, BaseFsm *> mStatusCollection;
public:
//命令一些實體去達到某個狀態(tài)
void Transaction(std::vector<FsmEntity *> & entities, const char * stateName);
//令某個實體達到某個狀態(tài)
void Transaction(FsmEntity * entity, const char * stateName);
//添加狀態(tài)機
void AddFsm( BaseFsm * fsm);
//刪除狀態(tài)機
void RemoveFsm(const char * stateName);
//獲取狀態(tài)機
BaseFsm * FindFsm(const char * stateName);
    //構(gòu)造
FsmManager(const char *fsname);
//析構(gòu)
virtual ~FsmManager();
};




State::State()
{
}
State::State(const char * name)
{
statename = name;
}

//構(gòu)造函數(shù)
FsmEntity::FsmEntity()
{
  mCurrentFsm = 0;
}
//析構(gòu)函數(shù)
FsmEntity::~FsmEntity()
{
  
}
//返回當前的狀態(tài)機
BaseFsm * FsmEntity::GetCurrentFsm()
{
return mCurrentFsm;
}
//狀態(tài)是否初始化了
bool FsmEntity::IsStateInited()
{
if(mCurrentFsm)
   return true;
else
   return false;
}
//設(shè)置當前狀態(tài)
void FsmEntity::InitState(BaseFsm *fsm)
{
if(mCurrentFsm == 0)
{
   mCurrentFsm = fsm;
}
else
{
  LOG(0, WARN_LV, "初始狀態(tài)已經(jīng)設(shè)定");
}
}
//保持狀態(tài)的方法
void FsmEntity::KeepState()
{
mCurrentFsm->Execute(this);
}
//改變狀態(tài)
void FsmEntity::ChangeState(BaseFsm *newFsm)
{
if(mCurrentFsm)
    //離開原來的狀態(tài)
    mCurrentFsm->Exit(this);
//設(shè)定現(xiàn)有狀態(tài)
mCurrentFsm = newFsm;
//進入現(xiàn)有狀態(tài)
mCurrentFsm->Enter(this);
//執(zhí)行現(xiàn)有的狀態(tài)
    mCurrentFsm->Execute(this);
}

//比較兩個狀態(tài)是否相同
bool State::operator ==(const State& other)
{
return statename == other.statename;
}

//進入狀態(tài)
void BaseFsm::Enter(FsmEntity *entity)
{
  LOG(0, DEBUG_LV, "進入%s狀態(tài)", state.statename.c_str());
}
//執(zhí)行狀態(tài)
void BaseFsm::Execute(FsmEntity *entity)
{
  LOG(0, DEBUG_LV, "執(zhí)行%s狀態(tài)", state.statename.c_str());
}
//離開狀態(tài)
void BaseFsm::Exit(FsmEntity *entity)
{
  LOG(0, DEBUG_LV, "離開%s狀態(tài)", state.statename.c_str());
}

bool BaseFsm::operator ==(const BaseFsm& other)
{
    return state == other.state;
}
//命令一些實體去達到某個狀態(tài)
void FsmManager::Transaction(std::vector<FsmEntity *> & entities, const char * stateName)
{
for(size_t i = 0; i < entities.size(); i ++)
{
  FsmEntity *entity = entities[i];
  Transaction(entity, stateName);
}
}
//令某個實體達到某個狀態(tài)
void FsmManager::Transaction(FsmEntity * entity, const char * stateName)
{
if(entity->GetCurrentFsm() && entity->GetCurrentFsm()->state.statename == stateName)
{
  entity->KeepState(); //保持之前的狀態(tài)
}
else
{
  BaseFsm * fsm = mStatusCollection[stateName];
  if(fsm)
  {
   //執(zhí)行狀態(tài)
   entity->ChangeState(fsm);
  }
  else
  {
            LOG(0, ERROR_LV, "找不到%s狀態(tài)", stateName);
  }
}
}

//添加狀態(tài)機
void FsmManager::AddFsm( BaseFsm * fsm)
{
  if(mStatusCollection.find(fsm->state.statename.c_str()) != mStatusCollection.end())
   return; //已經(jīng)添加過了
  //添加
  mStatusCollection[fsm->state.statename] = fsm;
}
//刪除狀態(tài)機
void FsmManager::RemoveFsm(const char * stateName)
{
  std::map<std::string, BaseFsm *>::iterator it = 0;
  if((it = mStatusCollection.find(stateName)) != mStatusCollection.end())
  {
     mStatusCollection.erase(it);
  }
}
//獲取狀態(tài)機
BaseFsm * FsmManager::FindFsm(const char * stateName)
{
std::map<std::string, BaseFsm *>::iterator it = 0;
if((it = mStatusCollection.find(stateName)) != mStatusCollection.end())
{
   BaseFsm * fsm = it->second;
   return fsm;
}
return 0;
}

//構(gòu)造
FsmManager::FsmManager(const char *fsname)
{
   name = fsname;
   LOG(0, DEBUG_LV, "構(gòu)造狀態(tài)機容器 %s", fsname);
}
//析構(gòu)
FsmManager::~FsmManager()
{
//移出所有的狀態(tài)機
for(std::map<std::string, BaseFsm *>::iterator it = mStatusCollection.begin(); it!= mStatusCollection.end(); it++)
{
       BaseFsm * fsm = it->second;
    if(fsm)
   delete fsm;
}
}



分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復(fù) 返回頂部 返回列表
主站蜘蛛池模板: 天天久久 | a视频在线观看 | 日本高清在线一区 | 一区二区三区欧美在线观看 | 9191成人精品久久 | 狼人伊人影院 | 国产精品中文字幕在线 | 国产精品入口 | 天天操一操 | 久久精品aaa | 美女黄视频网站 | 精品国产一区二区三区久久狼黑人 | 免费在线播放黄色 | 欧美一区二区在线观看 | 99久久婷婷国产综合精品电影 | 久久久久99 | 国产高潮av| 欧美国产日韩在线观看 | 亚洲国产欧美一区 | 巨大黑人极品videos精品 | 女朋友的闺蜜3韩国三级 | 久久综合久色欧美综合狠狠 | 久久久综合精品 | 久久久久久高潮国产精品视 | 久久精品亚洲精品国产欧美 | 情侣av | 成人国产精品免费观看 | 高清视频一区二区三区 | 手机av在线 | 成人精品在线视频 | 欧美日韩国产高清 | 日韩精品久久一区 | 欧美日韩在线视频一区 | 国产精品久久久久久久久久免费看 | 黑人巨大精品欧美一区二区一视频 | 中文字幕亚洲区一区二 | 色五月激情五月 | www.国产精品 | 91精品国产综合久久久亚洲 | 国产97人人超碰caoprom | 精品不卡 |