main.cc 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. // Example of world building, display, and successor computation for the artificial
  2. // intelligence path-finding lab
  3. //
  4. // Author: Didier LIME
  5. // Adapted by : Jovian HERSEMEULE
  6. // Date: 2018-10-03
  7. #include <iostream>
  8. #include <list>
  9. #include <cstdlib>
  10. #include <ctime>
  11. #include <stack>
  12. using namespace std;
  13. // Tile codes
  14. #define FREE 0
  15. #define WALL 1
  16. #define ORIGIN 2
  17. #define TARGET 3
  18. #define DISCOVERED 4
  19. #define TRACE 5
  20. unsigned int identifyTile(unsigned int y, unsigned int x, unsigned int l) {
  21. return y * l + x;
  22. }
  23. class World
  24. {
  25. private:
  26. // Number of columns
  27. unsigned int l;
  28. // Number of lines
  29. unsigned int h;
  30. // Size of the array
  31. const unsigned int size;
  32. // Unidimensional array for tiles
  33. int* board;
  34. public:
  35. // Constructor
  36. World(unsigned int l_, unsigned int h_, double p)
  37. :l(l_), h(h_), size(l_ * h_)
  38. {
  39. board = new int[size]();
  40. // Add walls to the first and last columns
  41. for (unsigned int i = 0; i < h; i++)
  42. {
  43. board[i * l] = WALL;
  44. board[i * l + l - 1] = WALL;
  45. }
  46. // Add walls to the first and last lines
  47. for (unsigned int j = 0; j < l; j++)
  48. {
  49. board[j] = WALL;
  50. board[(h - 1) * l + j] = WALL;
  51. }
  52. for (unsigned int i = 0; i < h; i++)
  53. {
  54. for (unsigned int j = 0; j < l; j++)
  55. {
  56. // add a wall in this tile with probability p and provided that it is neither
  57. // the starting tile nor the goal tile
  58. if ((double) rand() / RAND_MAX < p && !(i == 1 && j == 1) && !(i == h - 2 && j == l - 2))
  59. {
  60. board[i * l + j] = WALL;
  61. }
  62. }
  63. }
  64. }
  65. // Copy constructor
  66. World(const World& other)
  67. :l(other.l), h(other.h), size(other.size)
  68. {
  69. board = new int[size]();
  70. for (int k(0); k < size; k++) {
  71. board[k] = other.board[k];
  72. }
  73. }
  74. // Display the world
  75. void display()
  76. {
  77. for (unsigned int i = 0; i < h; i++)
  78. {
  79. for (unsigned int j = 0; j < l; j++)
  80. {
  81. switch (board[identifyTile(i, j, l)])
  82. {
  83. case FREE:
  84. cout << " ";
  85. break;
  86. case WALL:
  87. cout << "#";
  88. break;
  89. case ORIGIN:
  90. cout << "o";
  91. break;
  92. case TARGET:
  93. cout << "T";
  94. break;
  95. case DISCOVERED:
  96. cout << ":";
  97. break;
  98. case TRACE:
  99. cout << "+";
  100. break;
  101. }
  102. }
  103. cout << endl;
  104. }
  105. }
  106. // compute the successors of tile number i in world w
  107. // we return the number n of valid successors
  108. // the actual list is in array r where only the first n
  109. // elements are significant
  110. unsigned int successors(unsigned int i, unsigned int r[4])
  111. {
  112. unsigned int n = 0;
  113. if (i >= 0 && i < size && board[i] != WALL)
  114. {
  115. // if i is a correct tile number (inside the array and not on a wall)
  116. // look in the four adjacent tiles and keep only those with no wall
  117. const unsigned int moves[] = { i - 1, i + 1, i - l, i + l};
  118. for (unsigned int k = 0; k < 4; k++)
  119. {
  120. if (board[moves[k]] != WALL)
  121. {
  122. r[n] = moves[k];
  123. n++;
  124. }
  125. }
  126. }
  127. return n;
  128. }
  129. // Mark a list of points in the world
  130. void markAll(const list<unsigned int>& path, int value = TRACE) {
  131. for (auto tile : path) {
  132. markOne(tile, value);
  133. }
  134. }
  135. // Mark a point in the world
  136. void markOne(unsigned int tile, int value = ORIGIN) {
  137. board[tile] = value;
  138. }
  139. // Depth-first search
  140. // starting from tile number origin, find a path to tile number t
  141. // return true if such a path exists, false otherwise
  142. // if it exists the path is given in variable path (hence the reference &)
  143. const bool dfs(unsigned int origin, unsigned int target, list<unsigned int>& path, list<unsigned int>& discovered)
  144. {
  145. bool targetIsReached = false;
  146. bool explored[size];
  147. unsigned int previous[size];
  148. for (unsigned int k(0); k < size; k ++) {
  149. explored[k] = false;
  150. previous[k] = k;
  151. }
  152. explored[origin] = true;
  153. stack<unsigned int> open;
  154. open.push(origin);
  155. int current;
  156. int neighbour;
  157. unsigned int succs[4];
  158. unsigned int nbSuccs;
  159. do {
  160. current = open.top();
  161. open.pop();
  162. nbSuccs = successors(current, succs);
  163. for (unsigned int i(0); i < nbSuccs; i++) {
  164. neighbour = succs[i];
  165. if (!explored[neighbour]) {
  166. open.push(neighbour);
  167. previous[neighbour] = current;
  168. }
  169. }
  170. // Current tile is now processed
  171. explored[current] = true;
  172. discovered.push_back(current);
  173. // Stop if target found
  174. targetIsReached = (current == target);
  175. } while (!targetIsReached && !open.empty());
  176. // Build path
  177. if (targetIsReached) {
  178. do {
  179. path.push_back(current);
  180. current = previous[current];
  181. } while (current != origin);
  182. }
  183. return targetIsReached;
  184. }
  185. void showResults(bool exitFound, const list<unsigned int>& discovered, const list<unsigned int>& path) {
  186. markAll(discovered, DISCOVERED);
  187. if (!path.empty()) {
  188. markAll(path, TRACE);
  189. }
  190. display();
  191. // Display DFS results
  192. if (exitFound)
  193. cout << "SUCCESS !" << endl;
  194. else
  195. cout << "FAILURE ..." << endl;
  196. cout << discovered.size() << " tiles discovered;" << endl;
  197. cout << path.size() << " path length.";
  198. }
  199. };
  200. int main()
  201. {
  202. // Initialise the random number generator
  203. srand(time(0));
  204. // Create a world
  205. const unsigned int l(50), h(15);
  206. const double wallProbability(0.2);
  207. World w(l, h, wallProbability);
  208. unsigned int start(identifyTile(1, 1, l));
  209. unsigned int end(identifyTile(h - 2, l - 2, l));
  210. // Display it
  211. cout << endl << "Generated world" << endl;
  212. w.markOne(start, ORIGIN);
  213. w.markOne(end, TARGET);
  214. w.display();
  215. // Find a path with Depth-First Search
  216. World dfsWorld(w);
  217. list<unsigned int> dfsPath;
  218. list<unsigned int> dfsDiscovered;
  219. bool exitFound = dfsWorld.dfs(start, end, dfsPath, dfsDiscovered);
  220. // Display DFS
  221. cout << endl << "Depth-First Search" << endl;
  222. dfsWorld.showResults(exitFound, dfsDiscovered, dfsPath);
  223. // End
  224. return 0;
  225. }