• 6544阅读
  • 4回复

[转载][C++Qt]人工智能算法在游戏中演示 [复制链接]

上一主题 下一主题
离线jeffreylee
 

只看楼主 倒序阅读 楼主  发表于: 2013-04-16
转自:http://bbs.csdn.net/topics/390425195

人工智能是一个非常具有潜力的分支,顾名思义,它可以通过计算机指令模拟人的行为,在游戏中的人工智能就非常地多样了。对于FPS、TAB、RPG、STG、ADV等游戏,有着不同的人工智能,但都基于几种理论:有限状态机、遗传算法、神经网络等。下面我就采用游戏中最基础也是最常用的人工智能算法来进行演示。
我的开发环境是:
Windows: WindowsXP(SP3) + MinGW4.4/MinGW4.7 + Qt4.8.3/Qt5.0.1 + QtCreator2.6/QtCreator2.7
Linux: Ubuntu12.10 + gcc4.7 + Qt4.8.1/Qt4.8.4/Qt5.0.1 + QtCreator2.6/QtCreator2.7
演示程序下载地址:这里
源代码下载地址:这里
注意:由于Qt5.0.1中一个未知的Bug,编译运行后角色无法移动,所以请使用Qt4.6+来进行编译。
程序截图如下所示:




通过子窗口的控制选项,我们可以设置我们的初音ミク、镜音リン和镜音レン的控制方法,可以选择人工控制也可以选择AI控制。如果选择人工控制,那么可以通过按下“上下左右”键控制角色的移动,如果选择AI控制,那么角色会沿着场景作顺时针移动。
整个项目的代码量较多,我将选择有关人工智能的内容进行讲解,也希望大家沿着我的思路思考下去。
首先如何让角色沿着窗口作顺时针旋转?一个简单的想法就是:当角色将要达到窗口顶端那么角色将往右移动;角色将要到达窗口右端那么角色将往下移动,以此类推。按照这样的思路,我写了这样的AI代码:

  1. if ( m_pCharacter->pos( ).y( ) > 20.0 )  
  2.     {  
  3.         qDebug( "AI go Up." );  
  4.         m_pCharacter->SetAnimation( Character::_Up_ );  
  5.         emit TriggerTransition( );  
  6.     }  
  7.     else if ( m_pCharacter->pos( ).x( ) < 608.0 )  
  8.     {  
  9.         qDebug( "AI go Right." );  
  10.         m_pCharacter->SetAnimation( Character::_Right_ );  
  11.         emit TriggerTransition( );  
  12.     }  
  13.     else if ( m_pCharacter->pos( ).y( ) < 340.0 )  
  14.     {  
  15.         qDebug( "AI go Down." );  
  16.         m_pCharacter->SetAnimation( Character::_Down_ );  
  17.         emit TriggerTransition( );  
  18.     }  
  19.     else if ( m_pCharacter->pos( ).x( ) > 0.0 )  
  20.     {  
  21.         qDebug( "AI go Left." );  
  22.         m_pCharacter->SetAnimation( Character::_Left_ );  
  23.         emit TriggerTransition( );  
  24.     }



这也是我AI的第一个版本,但是正如QtMikuSnake7_ver_1应用程序截图中所示,它并不能达到应有的效果,角色一直在右上角打转。看来第一个版本有问题。问题在哪儿呢?这是由于我们将向上的判定优先于向下的判定,导致了角色在右上角处转至下后又转回了右上角。了解了这个问题之后第一个想法就是为向上判定添加约束条件,使其能够在右上角处正确地转至向下判定而不会折返。下面是AI的第二个版本:

  1. if ( m_pCharacter->pos( ).y( ) > 20.0 &&  
  2.          m_pCharacter->m_Direction != Character::_Down_ )  
  3.     {  
  4.         qDebug( "AI go Up." );  
  5.         m_pCharacter->SetAnimation( Character::_Up_ );  
  6.         emit TriggerTransition( );  
  7.     }  
  8.     else if ( m_pCharacter->pos( ).x( ) < 608.0 &&  
  9.               m_pCharacter->m_Direction != Character::_Left_ )  
  10.     {  
  11.         qDebug( "AI go Right." );  
  12.         m_pCharacter->SetAnimation( Character::_Right_ );  
  13.         emit TriggerTransition( );  
  14.     }  
  15.     else if ( m_pCharacter->pos( ).y( ) < 340.0 &&  
  16.               m_pCharacter->m_Direction != Character::_Up_ )  
  17.     {  
  18.         qDebug( "AI go Down." );  
  19.         m_pCharacter->SetAnimation( Character::_Down_ );  
  20.         emit TriggerTransition( );  
  21.     }  
  22.     else if ( m_pCharacter->pos( ).x( ) > 0.0 &&  
  23.               m_pCharacter->m_Direction != Character::_Right_ )  
  24.     {  
  25.         qDebug( "AI go Left." );  
  26.         m_pCharacter->SetAnimation( Character::_Left_ );  
  27.         emit TriggerTransition( );  
  28.     }  


为了保险,按照这种思路,将每一个方向判定都添加了约束条件,即判定当前的方向是何方向。按理说角色要往上走,那么当前的方向就肯定不是往下走,角色要往左走,当前的方向就肯定不是往右走。好了,运行一下,结果发现如QtMikuSnake7_ver_2所示的效果一样,角色在右下角处至左移了一格就往上走了。看来又是一次失误。



分析原因,发现向上判定的约束虽然解决了当前方向向下时仍进行向上判定的问题,可是未解决当前方向向左时仍然出现向上判定优先于向左判定的情况。看来还是需要再对向上判定进行进一步约束,按照“上右下左”的移动顺序,我们了解向上判定需要两个约束,向右判定需要一个约束,而向下判定不需要额外的约束,向左判定就更不需要了。下面AI的第三个版本:

  1. if ( m_pCharacter->pos( ).y( ) > 20.0 &&  
  2.          m_pCharacter->m_Direction != Character::_Down_ &&  
  3.          m_pCharacter->m_Direction != Character::_Left_ )  
  4.     {  
  5.         qDebug( "AI go Up." );  
  6.         m_pCharacter->SetAnimation( Character::_Up_ );  
  7.         emit TriggerTransition( );  
  8.     }  
  9.     else if ( m_pCharacter->pos( ).x( ) < 608.0 &&  
  10.               m_pCharacter->m_Direction != Character::_Left_ )  
  11.     {  
  12.         qDebug( "AI go Right." );  
  13.         m_pCharacter->SetAnimation( Character::_Right_ );  
  14.         emit TriggerTransition( );  
  15.     }  
  16.     else if ( m_pCharacter->pos( ).y( ) < 340.0 )  
  17.     {  
  18.         qDebug( "AI go Down." );  
  19.         m_pCharacter->SetAnimation( Character::_Down_ );  
  20.         emit TriggerTransition( );  
  21.     }  
  22.     else if ( m_pCharacter->pos( ).x( ) > 0.0 )  
  23.     {  
  24.         qDebug( "AI go Left." );  
  25.         m_pCharacter->SetAnimation( Character::_Left_ );  
  26.         emit TriggerTransition( );  
  27.     }  
  28.     else if ( m_pCharacter->pos( ).x( ) == 0.0 )// 为防止角色在左下角卡死设立的判断  
  29.     {  
  30.         m_pCharacter->SetDirection( Character::_Up_ );  
  31.     }  


注意到最下面一个判断,因为角色走在左下角的时候会因为都满足不了这些判定条件而陷入“卡死”状态,所以我们要进行“解锁”操作——将当前的方向设为向上,这样又可以满足向上的判定了。下面是程序QtMikuSnake7_ver_3的截图。



似乎这个问题圆满地解决了。但是我觉得这个代码还是写得太被动了,因为这些代码都是出了问题而一个一个地打补丁打上去的,非常被动。我们得换一个角度考虑。试想,如果一条语句能够“排队”,当让它执行的时候它排在最前面,执行完毕后它轮到最末尾,给下一条语句机会,要是这样的话,我们可以让上、右、下、左四条语句依次排队,一条语句一条语句地轮着运行。
这完全有可能实现!回想起来了吗?这不就是数据结构中经典的队列结构!可是,怎样才能让语句智能地移动到后面执行呢?这里需要使用一个类对这条语句进行封装。因为语句的格式相当有条理:if 判定条件 then 执行语句。我是这样封装的:

  1. class Clause: public QObject  
  2.     {  
  3.     public:  
  4.         Clause( Character* pParent = 0 ):  
  5.             QObject( pParent ), m_pCharacter( pParent ){ }  
  6.         virtual bool JudgeSentence( void ) = 0;  
  7.         virtual void Statement( void ) = 0;  
  8.     protected:  
  9.         Character*              m_pCharacter;  
  10.     };  


随后我设定一个队列,在Qt中有个现成的QQueue。

  1. QQueue<Clause*>         m_Clauses;  


接着进行语句类的定义,让其继承Clause类:

  1. class DirUpClause: public Clause  
  2.     {  
  3.     public:  
  4.         DirUpClause( Character* pParent = 0 ): Clause( pParent )  
  5.         {  
  6.       
  7.       
  8.         }  
  9.         bool JudgeSentence( void )  
  10.         {  
  11.             return m_pCharacter->pos( ).y( ) > 20.0;  
  12.         }  
  13.         void Statement( void )  
  14.         {  
  15.             qDebug( "AI go Up." );  
  16.             m_pCharacter->SetAnimation( Character::_Up_ );  
  17.         }  
  18.     };  
  19.       
  20.       
  21.     class DirDownClause: public Clause  
  22.     {  
  23.     public:  
  24.         DirDownClause( Character* pParent = 0 ): Clause( pParent )  
  25.         {  
  26.       
  27.       
  28.         }  
  29.         bool JudgeSentence( void )  
  30.         {  
  31.             return m_pCharacter->pos( ).y( ) < 340.0;  
  32.         }  
  33.         void Statement( void )  
  34.         {  
  35.             qDebug( "AI go Down." );  
  36.             m_pCharacter->SetAnimation( Character::_Down_ );  
  37.         }  
  38.     };  
  39.       
  40.       
  41.     class DirLeftClause: public Clause  
  42.     {  
  43.     public:  
  44.         DirLeftClause( Character* pParent = 0 ): Clause( pParent )  
  45.         {  
  46.       
  47.       
  48.         }  
  49.         bool JudgeSentence( void )  
  50.         {  
  51.             return m_pCharacter->pos( ).x( ) > 0.0;  
  52.         }  
  53.         void Statement( void )  
  54.         {  
  55.             qDebug( "AI go Left." );  
  56.             m_pCharacter->SetAnimation( Character::_Left_ );  
  57.         }  
  58.     };  
  59.       
  60.       
  61.     class DirRightClause: public Clause  
  62.     {  
  63.     public:  
  64.         DirRightClause( Character* pParent = 0 ): Clause( pParent )  
  65.         {  
  66.       
  67.       
  68.         }  
  69.         bool JudgeSentence( void )  
  70.         {  
  71.             return m_pCharacter->pos( ).x( ) < 608.0;  
  72.         }  
  73.         void Statement( void )  
  74.         {  
  75.             qDebug( "AI go Right." );  
  76.             m_pCharacter->SetAnimation( Character::_Right_ );  
  77.         }  
  78.     };  


最后我们在更新对象状态代码中进行一个简单地调用就可以了。

  1. foreach ( Clause* pClause, m_Clauses )  
  2.     {  
  3.         if ( pClause->JudgeSentence( ) )  
  4.         {  
  5.             pClause->Statement( );  
  6.             emit TriggerTransition( );  
  7.             break;  
  8.         }  
  9.         else  
  10.         {  
  11.             m_Clauses.dequeue( );  
  12.             m_Clauses.enqueue( pClause );  
  13.             break;  
  14.         }  
  15.     }  


上面的代码中,当条件满足的时候进行语句的执行,当条件不满足的时候将该语句从队列的头部移至队列的尾部。这样写虽然代码会比较多,但是思路清晰,对更复杂的状态维护起着重要的作用。执行起来效率也比较高,因为少了一些不必要的判定。下面是程序QtMikuSnake7_ver_4的截图。



上面的算法仅仅是一个很简单的演示,对于角色的运动还有诸如追击、自主规避、寻路等AI算法,对于复杂的游戏,状态的维护非常复杂且容易出错,而这个错误又不像程序宕机那样容易觉察,这时需要一个特定的职位——脚本设计师来解决此类问题。脚本设计师面对着各类的数据,通过自身熟练的脚本语言来对程序的各类参数进行微调,一款成功的游戏总有脚本设计师付出的辛勤汗水。所以说脚本设计师都是艺术家。
后记:程序QtMikuSnake7对上一个版本的帧框大小进行了修正,刷新不会出现残影的现象,有关AI测试代码也将在下一个版本中被去除。  


离线toby520

只看该作者 1楼 发表于: 2013-04-16
Lee发的帖子根本没法阅读呢,能否修改修改
QtQML多多指教开发社区 http://qtclub.heilqt.com
将QtCoding进行到底
关注移动互联网,关注金融
开发跨平台客户端,服务于金融行业
专业定制界面
群号:312125701   373955953(qml控件定做)
离线makai
只看该作者 2楼 发表于: 2013-04-16
回 1楼(toby520) 的帖子
估计是技术性太强了吧!嘿嘿
编程改变生活
离线XChinux

只看该作者 3楼 发表于: 2013-04-16
这回能看了吧。
二笔 openSUSE Vim N9 BB10 XChinux@163.com 网易博客 腾讯微博
承接C++/Qt、Qt UI界面、PHP及预算报销系统开发业务
离线nmiirq

只看该作者 4楼 发表于: 2013-04-17
图裂了。。。
快速回复
限100 字节
 
上一个 下一个