#実行したいこと発生している問題
C++で壁や箱,ゴールの数がどのような設定でも動く倉庫番を作成しているのですが,コンパイルは上手くいったものの,正しく動きません(具体的にはゴールが1つしか表示されない,それによって1つの箱をゴールに動かしたら終わってしまう,5×5のマスが整っていないといった問題があります)。詳しい原因が分からないので教えていただきたいです。print()関数,completed()関数,move()関数を特に確認してほしいです。また,STLアルゴリズムの扱いに慣れていないので扱い方についても教えてほしいです。
ちなみに,以下の要領でプログラムを作成しようと考えています。
①#include<vector>を利用し「ゲームフィールドの初期設定」部分を以下のようにする。
const int row{3}; const int column{3}; const std::vector<Pos> wall{}; //壁0 個 const std::vector<Pos> goal{{0,0},{2,1}}; //ゴール2 個 Pos player{2,0}; std::vector<Pos> box{{0,1},{1,1}}; //箱2 個
②#include<algorithm>を基に、wall, box, goal が「vectorの要素に含まれているかどうかを判定する」ようにする。
③completed 関数を「順番に関係なくgoalの全ての要素がboxの全ての要素と一致しているかを判定する」ものにする。
④box の移動時に、他のbox に重なるかどうかの判定を追加し、重なっていない場合のみ移動するように変更する。
結果
C++
1XXXXX 2X+ $ X 3X $ X 4X@ X 5XXXXX 6 7r 8XXXXX 9X+ $ X 10X $ X 11X @ X 12XXXXX 13 14r 15XXXXX 16X+ $ X 17X $ X 18X @ X 19XXXXX 20 21r 22XXXXX 23X+ $ X 24X $ X 25X @ X 26XXXXX 27 28u 29XXXXX 30X+ $ X 31X $ @ X 32X X 33XXXXX 34 35u 36XXXXX 37X+ $ @ X 38X $ X 39X X 40XXXXX 41 42u 43XXXXX 44X+ $ @ X 45X $ X 46X X 47XXXXX 48 49l 50XXXXX 51XO@ X 52X $ X 53X X 54XXXXX 55 56The work is completed!! 57
##プログラム
C++
1#include<iostream> 2#include<vector> 3#include<algorithm> 4 5//i行j列の位置座標の定義 6struct Pos { 7 int i, j; 8 bool operator==(const Pos p) const { return i == p.i && j == p.j;}; 9 bool operator!=(const Pos p) const { return i != p.i || j != p.j;}; 10}; 11 12//倉庫番のゲームフィールドの定義 13class Field { 14private: 15 //ゲームフィールドの初期設定 16 const int row{3}; //行数(縦の長さ) 17 const int column{3}; //列数(横の長さ) 18 const std::vector<Pos> wall{}; //壁0個 19 const std::vector<Pos> goal{{0,0},{2,1}}; //ゴール2個 20 Pos player{2,0}; //プレイヤーの初期位置 21 std::vector<Pos> box{{0,1},{1,1}}; //箱2個 22 23public: 24 //フィールドを端末上に表示 25 void print(){ 26 for(int i = -1; i <= row; i++){ 27 for(int j = -1; j <= column; j++){ 28 Pos p{i,j}; 29 //フィールドの周囲に壁を表示 30 if(i < 0 || i >= row || j < 0 || j >= column){ 31 std::cout << 'X'; 32 } 33 //プレイヤー、壁、ゴール、箱を表示 34 //プレイヤーの場所では'@'を表示 35 //壁の場所では'X'を表示 36 //ゴールと箱が重なっている場合は'O'を表示 37 //ゴールのみの場合は'+'を表示 38 //箱のみの場合は'$'を表示 39 //それ以外は空白' 'を表示 40 else { 41 if(p == player) {std::cout << '@';} 42 if(p == goal[i] && p != box[i]) {std::cout << '+';} 43 if(p == box[i] && p != goal[i]) {std::cout << '$';} 44 if(p == goal[i] && p == box[i]) {std::cout << 'O';} 45 else {std::cout << ' ';} 46 } 47 } 48 std::cout << '\n'; 49 } 50 std::cout << '\n'; 51 } 52 53 //順番に関係なくゴールと箱が重なっている場合はゲームクリアと判定 54 bool completed(){ 55 for (int a = 0; a < goal.size(); a++) { 56 if(goal[a] == box[a]) 57 return true; 58 else 59 return false; 60 } 61 } 62 63 //u(up), d(down), r(right), l(left)の文字列を受けてプレイヤーをその方向に移動 64 //move関数の引数として、Pos型引数を移動方向に変更するラムダ式を与える 65 //例えば d=='u'の場合は、Pos型変数pを引数として、p.iを1減らして、pを返すラムダ式を与える 66 void movec(char d){ 67 if(d == 'u') move([](Pos p){--p.i;return p;}); 68 else if(d == 'd') move([](Pos p){++p.i;return p;}); 69 else if(d == 'r') move([](Pos p){++p.j;return p;}); 70 else if(d == 'l') move([](Pos p){--p.j;return p;}); 71 } 72 73 //Pos型変数を移動する関数nextを受け取りフィールドの状態更新を行う 74 template<typename T> 75 void move(T next) { 76 //Pos型引数pが移動可能な位置か判定する関数をラムダ式で定義 77 //pが0~(row-1), 0~(column-1)の座標に収まっているか、他の箱と重なっていないかを判定 78 auto can_move = [=](Pos p){ 79 for (int a=0; a < box.size()-1; a++) { 80 return (0 <= p.i && p.i <= row-1) && (0 <= p.j && p.j <= column-1) && box[a+1] != box[a]; 81 } 82 }; 83 84 Pos pnext = next(player); //プレイヤーの次の位置の計算 85 if(can_move(pnext)){ //プレイヤーが移動可能な場合のみ更新 86 //プレイヤーが箱に重ならない場合はそのまま移動 87 for (int a = 0; a < box.size(); a++) { 88 if(pnext != box[a]) { 89 player = pnext; 90 } 91 //プレイヤーが箱に重なる場合は箱の移動判定が必要 92 else { 93 Pos bnext = next(box[a]);//箱の次の位置を計算 94 //箱が移動可能な場合はプレイヤーと箱を移動 95 if(can_move(bnext)) 96 player = pnext; 97 box[a] = bnext; 98 } 99 } 100 } 101 } 102}; 103 104 105int main() { 106 Field f; //フィールドの初期化 107 char c; 108 f.print(); //初期フィールドの表示 109 while(std::cin >> c) { 110 // 'q' の場合は終了(リタイア) 111 if(c == 'q'){ 112 std::cout << "quit" << '\n'; 113 exit(1); 114 } 115 // 'u', 'd', 'r', 'l' の場合はその移動方向に従って更新し、ゲームクリアの場合は終了 116 if(c == 'u' || c == 'd' || c == 'r' || c == 'l'){ 117 f.movec(c); 118 f.print(); 119 if(f.completed()){ 120 std::cout << "The work is completed!!\n"; 121 break; 122 } 123 } 124 //上記以外の入力の場合は正しい入力候補を表示 125 else std::cout << "u(up), d(down), r(right), l(left), q(quit)" << '\n'; 126 } 127}
##試したこと
構造体のvectorを扱うため,それぞれの箱やゴール,壁が条件を満たすかを判定できるためにfor文を使って作成しました。
##補足
C++17を利用しています。
回答3件
あなたの回答
tips
プレビュー