B-Teck!

お仕事からゲームまで幅広く

【JavaScript】相手を追尾するAIを考える その1

はじめに

この記事にこんなコメントがつきました。
beatdjam.hatenablog.com

コメント失礼します。 ゲーム開発を学んでいるプログラマーの卵です。

敵AIの追尾について勉強していて、 縦横マス目移動でプレイヤーを追尾出来るような敵の処理にしたいのですが、中々上手くいっていない状態です。

どのようにしたら縦横マス目移動で追尾してくれますでしょうか。 よろしくお願いします。

今回は状況がわからないため、一番簡単な障害物のないXY平面上でPlayerを追尾するEnemyという想定で、
どのように考えていけばいいかを説明していきたいと思います。

追尾の考え方

追尾というのは難しく考えてしまいがちですが、実質「現在の状態から相手に1マス近づく」という行動の繰り返しです。

f:id:beatdjam:20160627202855p:plain

1マスずつ近づくために必要な状況を切り分けて見ると、どう実装するべきかがわかってきます。

問題の切り分け

さて、実際どう実装するかを考えていきましょう。
X座標、Y座標を近づけたい場合は下記の3つの状況が考えられると思います。
f:id:beatdjam:20160627205039p:plain
X座標・Y座標両方があっていない場合は、今回はランダムでどちらかを選択するようにしてみましょう。

実装

今回はJavaScriptで実装してみました。動かすのが楽だったので。
moveEnemyとgetMovePosが今回の解説部分です。
コンソールに、Playerに近づいていく様子が出力されます。

おわりに

ざっくりとした説明でしたが、雰囲気だけでも掴んでいただけたでしょうか。
しかし、今回の実装ではマップ上に障害物や進めないエリアがあった場合に簡単に動けなくなってしまいます。
この問題を解決するには、経路探索と呼ばれるロジックを実装する必要があるのですが…、少し長くなってしまうので次回以降に記事にできたらいいなと思います。
基本の考え方として、「相手の位置を確定する」、「自分がどう移動するかを考える」という点は変わらないので、参考になれば幸いです。

最後に一応こちらにもソースを置いておきます。

"use strict"
process.stdin.resume();
process.stdin.setEncoding('utf8');

/*
* 設定値エリア(色々変えてみてください)
*/
//player、enemyそれぞれにx,y座標を設定する
var player = {"x":1,"y":1};
var enemy = {"x":4,"y":3};

//移動回数
var count = 6;

//マップの広さ
var xmax = 4;
var ymax = 4;

/*
* 実行
*/
//enemyを移動
for(var i = 0;i < count;i++){
  //現在のマップ出力
  renderingMap(player,enemy,xmax,ymax);  
  //enemyを動かす
  enemy = moveEnemy(player,enemy);
}

/*
* 関数定義
*/
//playerの座標に応じてenemyを移動させる
function moveEnemy(player,enemy){
  //playerとenemyの距離差分を取得する
  var xDistance = enemy.x - player.x;
  var yDistance = enemy.y - player.y;
  
  //playerとenemyに距離がある場合
  if(xDistance === 0){
    //x座標が同じ場合y座標のみ移動
    enemy.y = getMovePos(yDistance, enemy.y);
  }else if(yDistance === 0){
    //y座標が同じ場合x座標のみ移動
    enemy.x = getMovePos(xDistance, enemy.x);
  }else{
    //どちらも差がある場合、ランダムでどちらか1座標移動する
    var rand = Math.floor( Math.random() * 2);
    if(rand === 0){
      enemy.y = getMovePos(yDistance, enemy.y);      
    }else{
      enemy.x = getMovePos(xDistance, enemy.x);        
    }
  }
  
  return enemy;
}

//playerの位置関係によって移動方向を分け、移動先を返す
function getMovePos(distance, pos){
    if(distance > 0){
      pos--;
    }else{
      pos++;        
    }     
    return pos;  
}

//map描画
function renderingMap(player, enemy,xmax,ymax){
    for(var j = 1;j <= ymax;j++){
        var tmpStr = "";
        for(var k = 1;k <= xmax;k++){
            if(player.x === k && player.y === j){
                tmpStr += "p";
            }else if(enemy.x === k && enemy.y === j){
                tmpStr += "e";
            }else{
                tmpStr += "*";
            }
        }
        console.log(tmpStr);  
    }
    console.log(" ");
}

【PHP】標準入出力を行うクラス

<?php
    class inputManager{
        private $index = 0;
        private $input = [];
        //区切り文字設定
        const SPRIT_STR = " ";
        
        /**
         * コンストラクタ
         * 
         * @access public
         * @param  なし
         * @return なし
         */
        public function __construct($testFlg = false, $testInput = null){
            if(!$testFlg){
                    $input_lines = trim(fgets(STDIN));
                    $input_lines = str_replace(array("\r\n","\r","\n"), "", $input_lines);
                    do {
                        $this->input[] = $input_lines;
                        $input_lines = trim(fgets(STDIN));
                        $input_lines = str_replace(array("\r\n","\r","\n"), "", $input_lines);
                    } while ($input_lines !== "");
            }else{
                $this->input = $testInput;
            }
        }
        
        /**
         * value
         * 現在のインデックスの値を取得する
         * 
         * @access public
         * @param  なし
         * @return string|null 現在のインデックスの値
         */
        public function value(){
            if($this->index < $this->length()){
                return $this->input[$this->index];
            }else{
                return null;
            }
        }
        
        /**
         * explodeValue
         * 現在のインデックスの値をexplodeした配列を取得する
         * 
         * @access public
         * @param  なし
         * @return Array|null explode後配列
         */
        public function explodeValue(){
            if($this->index < $this->length()){
                return explode(self::SPRIT_STR, $this->input[$this->index]);
            }else{
                return null;
            }
        }
        
        /**
         * setindex
         * インデックスを設定する
         * 
         * @access public
         * @param  $i_index 設定するインデックス
         * @return $this
         */
        public function setindex(int $i_index){
            $this->index = $i_index;
            return $this;
        }
        
        /**
         * next
         * インデックスを1つすすめる
         * 
         * @access public
         * @param  なし
         * @return $this
         */
        public function next(){
            $this->index++;
            return $this;
        }
        
        /**
         * length
         * 入力の長さを取得
         * 
         * @access public
         * @param  なし
         * @return int $this->input 配列の長さ
         */
        public function length(){
            return count($this->input);
        }
    }
    
    class out{
        /**
         * output
         * 引数を出力
         * 入力がない場合は空行を出力する
         * 
         * @access public
         * @param  string $i_val 出力する文字列
         * @return なし
         */
        public static function output(string $i_val = ""){
            echo $i_val. "\n";
        }
    }
    
    //テスト用入力
    $input[] = "123";
    $input[] = "456";
    $input[] = "789";
    $input[] = "aaa";
    $input[] = "bbb";
    
    $io = new inputManager(true,$input);
    //長さの取得
    out::output($io->length());
    
    //
    out::output();
    
    //インデックス設定の確認
    out::output($io->value());
    $io->next();
    out::output($io->value());    
    $io->setindex(2);
    out::output($io->value());    
    out::output($io->next()->value());   
    out::output($io->next()->value());   
    
    out::output();
    
    //入力を末尾まで出力する
    $io->setindex(0);
    while(!is_null($io->value())){
        out::output($io->value());
        $io->next();
    }
    
    out::output();
    
    //インデックスをセットして末尾まで出力する
    for($i = 0;$i < $io->length(); $i++){
        out::output($io->setindex($i)->value());
    }

【PHP】連想配列とかオブジェクトのソート

多次元連想配列やオブジェクトを詰めた配列はusort関数でいい感じにソートする。
PHP: usort - Manual

多次元連想配列のソートをする場合

<?php
    //連想配列でのソート
    $test = [["price" => "30", "stock" => "100"],
             ["price" => "20", "stock" => "120"],
             ["price" => "10", "stock" => "115"]];
    
    //ソート
    usort($test, function ($a, $b) {
        return $a['price'] < $b['price'] ? -1 : 1;
        //逆順の場合はこっち
        //return  $a['price'] > $b['price'] ? -1 : 1;
    });    

    print_r($test);

オブジェクトを詰めた配列のソートをする場合

<?php
    class testClass{
        public $price;
        public $stock;
                
        function __construct($price,$stock){
            $this->price = $price;
            $this->stock = $stock;
        }
    }
    
    $test2 = [];
    $test2[] = new testClass("30","100");
    $test2[] = new testClass("20","120");
    $test2[] = new testClass("10","115");    
    
    usort($test2, function ($a, $b) {
        return $a->stock< $b->stock? -1 : 1;
    });    

    print_r($test2);