游戏状态机的设计与实现

前言:
     游戏编程中对状态机的理解和应用,是体现程序员是否对游戏编程入门的重要指标。本篇文章描述状态机的原理,以及如何实现。并探讨状态机的扩展性和易用性。


什么是状态机:
     1、状态机是通过状态变量来描述不同状态
     2、状态机变量是互斥的
     3、状态机的分割是状态机好坏的标准

状态机的好处:
     1、降低整个系统的复杂性
     2、容易扩展
     3、容易维护

如何实现状态机:
     1、通过不同的状态分割逻辑
     2、通过面向对象思想来扩展和分割逻辑

状态机简单类型:

1、定义状态机类型

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. enum PlayerState{  
  2.     INVALID,  
  3.     STAND,  
  4.     MOVE,  
  5.     ATTACK,  
  6.     DIE  
  7. };  

2、实现更新状态,在不同的状态执行不同的逻辑

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void Player::Update(float ts){  
  2.     switch(user_state_){  
  3.         case STAND:  
  4.             Stand(ts);  
  5.             break;  
  6.         case MOVE:  
  7.             Move(ts);  
  8.             break;  
  9.         case ATTACK:  
  10.             Attack(ts);  
  11.             break;  
  12.         case DIE:  
  13.             Die();  
  14.             return;  
  15.         default:  
  16.             std::cout<<"error\n";  
  17.   
  18.     }  
  19.   
  20.     if(hp_ <= 0){  
  21.         SetState(DIE);  
  22.     }  
  23. }  
3、切换状态,在切换状态的时候做一些事情

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void Player::SetState(PlayerState state){  
  2.     if(state == user_state_){  
  3.         return;  
  4.     }  
  5.     switch(state){  
  6.         case STAND:  
  7.             std::cout << "----begin stand--------\n";  
  8.             break;  
  9.         case MOVE:  
  10.             std::cout << "----begin move--------\n";  
  11.             break;  
  12.         case ATTACK:  
  13.             std::cout << "----begin attack--------\n";  
  14.             break;  
  15.         case DIE:  
  16.             std::cout << "----begin die--------\n";  
  17.             break;  
  18.         default:  
  19.             std::cout <<"the state is error";  
  20.             break;  
  21.     }  
  22.     user_state_ = state;  
  23. }  

     这种状态机小而精悍,如果在一个对象中有很多标志量来标记实例的状态,这时候该考虑下通过这种小型的状态机来实现了。但是这种状态机如果状态变量比较多,扩展性并不好。并且复杂性会随着状态机的增多,指数型增加。整个编译单元的代码量也会很大,对易读性和维护性都是负面影响。

状态机面向对象类型:
     面向对象类的状态机是一种更容易扩展的新型状态机,通过单间实现方式,使用更少的内存,先看下整个状态机的uml设计图。



  首先是通过接口定义通用状态机接口,然后定义了单间的接口。这种方式统一让所有的状态实现三个函数,这三个函数分别对应切入状态,在状态中,退出状态,需要执行的逻辑分别放在这三个函数里执行,通过这样的分割,状态很容易扩展,也不会混乱。具体代码实现,请看下面说面里面github的地址。
     在StateManager是专门管理角色状态的管理类,每个角色对象包含一个状态机管理类。

总结:
     状态机的模型是非常简单,但并不是每个人都能设计好的状态机。因为好的状态机不仅需要对程序的把握要比较到位,同时需要对整个业务的理解比较到位。好的状态机使程序变的更加简洁,易扩展,容易查找bug,还非常稳定。坏得状态分割只会让程序晦涩难懂。

说明:
1、通过两个状态机实现了两个简单的猜数打怪兽游戏。
2、所有完整程序都可以到 这个地址查看,下载,修改。
3、整个代码都是通过C++ 完成的,编译环境是osx 10.10 + LLVM 6.0 , C++使用 -std=c++1y。程序写了makefile,所以如果在其他平台只需要简单修改下makefile就可以快乐的玩耍了。

什么是状态机:
     1、状态机是通过状态变量来描述不同状态
     2、状态机变量是互斥的
     3、状态机的分割是状态机好坏的标准

状态机的好处:
     1、降低整个系统的复杂性
     2、容易扩展
     3、容易维护

如何实现状态机:
     1、通过不同的状态分割逻辑
     2、通过面向对象思想来扩展和分割逻辑

状态机简单类型:

1、定义状态机类型

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. enum PlayerState{  
  2.     INVALID,  
  3.     STAND,  
  4.     MOVE,  
  5.     ATTACK,  
  6.     DIE  
  7. };  

2、实现更新状态,在不同的状态执行不同的逻辑

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void Player::Update(float ts){  
  2.     switch(user_state_){  
  3.         case STAND:  
  4.             Stand(ts);  
  5.             break;  
  6.         case MOVE:  
  7.             Move(ts);  
  8.             break;  
  9.         case ATTACK:  
  10.             Attack(ts);  
  11.             break;  
  12.         case DIE:  
  13.             Die();  
  14.             return;  
  15.         default:  
  16.             std::cout<<"error\n";  
  17.   
  18.     }  
  19.   
  20.     if(hp_ <= 0){  
  21.         SetState(DIE);  
  22.     }  
  23. }  
3、切换状态,在切换状态的时候做一些事情

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. void Player::SetState(PlayerState state){  
  2.     if(state == user_state_){  
  3.         return;  
  4.     }  
  5.     switch(state){  
  6.         case STAND:  
  7.             std::cout << "----begin stand--------\n";  
  8.             break;  
  9.         case MOVE:  
  10.             std::cout << "----begin move--------\n";  
  11.             break;  
  12.         case ATTACK:  
  13.             std::cout << "----begin attack--------\n";  
  14.             break;  
  15.         case DIE:  
  16.             std::cout << "----begin die--------\n";  
  17.             break;  
  18.         default:  
  19.             std::cout <<"the state is error";  
  20.             break;  
  21.     }  
  22.     user_state_ = state;  
  23. }  

     这种状态机小而精悍,如果在一个对象中有很多标志量来标记实例的状态,这时候该考虑下通过这种小型的状态机来实现了。但是这种状态机如果状态变量比较多,扩展性并不好。并且复杂性会随着状态机的增多,指数型增加。整个编译单元的代码量也会很大,对易读性和维护性都是负面影响。

状态机面向对象类型:
     面向对象类的状态机是一种更容易扩展的新型状态机,通过单间实现方式,使用更少的内存,先看下整个状态机的uml设计图。



  首先是通过接口定义通用状态机接口,然后定义了单间的接口。这种方式统一让所有的状态实现三个函数,这三个函数分别对应切入状态,在状态中,退出状态,需要执行的逻辑分别放在这三个函数里执行,通过这样的分割,状态很容易扩展,也不会混乱。具体代码实现,请看下面说面里面github的地址。
     在StateManager是专门管理角色状态的管理类,每个角色对象包含一个状态机管理类。

总结:
     状态机的模型是非常简单,但并不是每个人都能设计好的状态机。因为好的状态机不仅需要对程序的把握要比较到位,同时需要对整个业务的理解比较到位。好的状态机使程序变的更加简洁,易扩展,容易查找bug,还非常稳定。坏得状态分割只会让程序晦涩难懂。

说明:
1、通过两个状态机实现了两个简单的猜数打怪兽游戏。
2、所有完整程序都可以到 这个地址查看,下载,修改。
3、整个代码都是通过C++ 完成的,编译环境是osx 10.10 + LLVM 6.0 , C++使用 -std=c++1y。程序写了makefile,所以如果在其他平台只需要简单修改下makefile就可以快乐的玩耍了。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值