main.cc 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  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. #include <queue>
  13. using namespace std;
  14. // Tile codes
  15. #define FREE 0
  16. #define WALL 1
  17. #define ORIGIN 2
  18. #define TARGET 3
  19. #define DISCOVERED 4
  20. #define TRACE 5
  21. unsigned int identifyTile(unsigned int y, unsigned int x, unsigned int l) {
  22. return y * l + x;
  23. }
  24. class World
  25. {
  26. private:
  27. // Number of columns
  28. unsigned int l;
  29. // Number of lines
  30. unsigned int h;
  31. // Size of the array
  32. const unsigned int size;
  33. // Unidimensional array for tiles
  34. int* board;
  35. public:
  36. // Constructor
  37. World(unsigned int l_, unsigned int h_, double p)
  38. :l(l_), h(h_), size(l_ * h_)
  39. {
  40. board = new int[size]();
  41. // Add walls to the first and last columns
  42. for (unsigned int i = 0; i < h; i++)
  43. {
  44. board[i * l] = WALL;
  45. board[i * l + l - 1] = WALL;
  46. }
  47. // Add walls to the first and last lines
  48. for (unsigned int j = 0; j < l; j++)
  49. {
  50. board[j] = WALL;
  51. board[(h - 1) * l + j] = WALL;
  52. }
  53. for (unsigned int i = 0; i < h; i++)
  54. {
  55. for (unsigned int j = 0; j < l; j++)
  56. {
  57. // add a wall in this tile with probability p and provided that it is neither
  58. // the starting tile nor the goal tile
  59. if ((double) rand() / RAND_MAX < p && !(i == 1 && j == 1) && !(i == h - 2 && j == l - 2))
  60. {
  61. board[i * l + j] = WALL;
  62. }
  63. }
  64. }
  65. }
  66. // Copy constructor
  67. World(const World& other)
  68. :l(other.l), h(other.h), size(other.size)
  69. {
  70. board = new int[size]();
  71. for (int k(0); k < size; k++) {
  72. board[k] = other.board[k];
  73. }
  74. }
  75. // Display the world
  76. void display()
  77. {
  78. for (unsigned int i = 0; i < h; i++)
  79. {
  80. for (unsigned int j = 0; j < l; j++)
  81. {
  82. switch (board[identifyTile(i, j, l)])
  83. {
  84. case FREE:
  85. cout << " ";
  86. break;
  87. case WALL:
  88. cout << "#";
  89. break;
  90. case ORIGIN:
  91. cout << "o";
  92. break;
  93. case TARGET:
  94. cout << "T";
  95. break;
  96. case DISCOVERED:
  97. cout << ":";
  98. break;
  99. case TRACE:
  100. cout << "+";
  101. break;
  102. }
  103. }
  104. cout << endl;
  105. }
  106. }
  107. // compute the successors of tile number i in world w
  108. // we return the number n of valid successors
  109. // the actual list is in array r where only the first n
  110. // elements are significant
  111. unsigned int successors(unsigned int i, unsigned int r[4])
  112. {
  113. unsigned int n = 0;
  114. if (i >= 0 && i < size && board[i] != WALL)
  115. {
  116. // if i is a correct tile number (inside the array and not on a wall)
  117. // look in the four adjacent tiles and keep only those with no wall
  118. const unsigned int moves[] = { i - 1, i + 1, i - l, i + l};
  119. for (unsigned int k = 0; k < 4; k++)
  120. {
  121. if (board[moves[k]] != WALL)
  122. {
  123. r[n] = moves[k];
  124. n++;
  125. }
  126. }
  127. }
  128. return n;
  129. }
  130. // Mark a list of points in the world
  131. void markAll(const list<unsigned int>& path, int value = TRACE) {
  132. for (auto tile : path) {
  133. markOne(tile, value);
  134. }
  135. }
  136. // Mark a point in the world
  137. void markOne(unsigned int tile, int value = ORIGIN) {
  138. board[tile] = value;
  139. }
  140. // Depth-first search
  141. // starting from tile number origin, find a path to tile number t
  142. // return true if such a path exists, false otherwise
  143. // if it exists the path is given in variable path (hence the reference &)
  144. const bool dfs(unsigned int origin, unsigned int target, list<unsigned int>& path, list<unsigned int>& discovered)
  145. {
  146. bool targetIsReached = false;
  147. bool explored[size];
  148. unsigned int previous[size];
  149. for (unsigned int k(0); k < size; k ++) {
  150. explored[k] = false;
  151. previous[k] = k;
  152. }
  153. explored[origin] = true;
  154. stack<unsigned int> open;
  155. open.push(origin);
  156. int current;
  157. int neighbour;
  158. unsigned int succs[4];
  159. unsigned int nbSuccs;
  160. do {
  161. current = open.top();
  162. open.pop();
  163. nbSuccs = successors(current, succs);
  164. for (unsigned int i(0); i < nbSuccs; i++) {
  165. neighbour = succs[i];
  166. if (!explored[neighbour]) {
  167. open.push(neighbour);
  168. explored[neighbour] = true;
  169. previous[neighbour] = current;
  170. }
  171. }
  172. // Current tile is now processed
  173. discovered.push_back(current);
  174. // Stop if target found
  175. targetIsReached = (current == target);
  176. } while (!targetIsReached && !open.empty());
  177. // Remove origin and target
  178. if (!discovered.empty() && discovered.front() == origin)
  179. discovered.pop_front();
  180. if (!discovered.empty() && discovered.back() == target)
  181. discovered.pop_back();
  182. // Build path
  183. if (targetIsReached) {
  184. do {
  185. path.push_back(current);
  186. current = previous[current];
  187. } while (current != origin);
  188. path.pop_front();
  189. }
  190. return targetIsReached;
  191. }
  192. // Breadth-first search
  193. // starting from tile number origin, find a path to tile number t
  194. // return true if such a path exists, false otherwise
  195. // if it exists the path is given in variable path (hence the reference &)
  196. const bool bfs(unsigned int origin, unsigned int target, list<unsigned int>& path, list<unsigned int>& discovered)
  197. {
  198. bool targetIsReached = false;
  199. bool explored[size];
  200. unsigned int previous[size];
  201. for (unsigned int k(0); k < size; k ++) {
  202. explored[k] = false;
  203. previous[k] = k;
  204. }
  205. explored[origin] = true;
  206. queue<unsigned int> open;
  207. open.push(origin);
  208. int current;
  209. int neighbour;
  210. unsigned int succs[4];
  211. unsigned int nbSuccs;
  212. do {
  213. current = open.front();
  214. open.pop();
  215. nbSuccs = successors(current, succs);
  216. for (unsigned int i(0); i < nbSuccs; i++) {
  217. neighbour = succs[i];
  218. if (!explored[neighbour]) {
  219. open.push(neighbour);
  220. explored[neighbour] = true;
  221. previous[neighbour] = current;
  222. }
  223. }
  224. // Current tile is now processed
  225. discovered.push_back(current);
  226. // Stop if target found
  227. targetIsReached = (current == target);
  228. } while (!targetIsReached && !open.empty());
  229. // Remove origin and target
  230. if (!discovered.empty()) {
  231. discovered.pop_front();
  232. if (!discovered.empty())
  233. discovered.pop_back();
  234. }
  235. // Build path
  236. if (targetIsReached) {
  237. do {
  238. path.push_back(current);
  239. current = previous[current];
  240. } while (current != origin);
  241. path.pop_front();
  242. }
  243. return targetIsReached;
  244. }
  245. void showResults(bool exitFound, const list<unsigned int>& discovered, const list<unsigned int>& path) {
  246. markAll(discovered, DISCOVERED);
  247. if (!path.empty()) {
  248. markAll(path, TRACE);
  249. }
  250. display();
  251. // Display DFS results
  252. if (exitFound)
  253. cout << "SUCCESS !" << endl;
  254. else
  255. cout << "FAILURE ..." << endl;
  256. cout << discovered.size() << " tiles discovered;" << endl;
  257. cout << path.size() << " path length." << endl;
  258. }
  259. };
  260. int main()
  261. {
  262. // Initialise the random number generator
  263. srand(time(0));
  264. // Create a world
  265. const unsigned int l(50), h(15);
  266. const double wallProbability(0.2);
  267. World w(l, h, wallProbability);
  268. unsigned int start(identifyTile(1, 1, l));
  269. unsigned int end(identifyTile(h - 2, l - 2, l));
  270. // Display it
  271. cout << endl << "Generated world" << endl;
  272. w.markOne(start, ORIGIN);
  273. w.markOne(end, TARGET);
  274. w.display();
  275. // 1
  276. cout << endl << "Depth-First Search" << endl;
  277. World dfsWorld(w);
  278. list<unsigned int> dfsPath;
  279. list<unsigned int> dfsDiscovered;
  280. bool dfsExitFound = dfsWorld.dfs(start, end, dfsPath, dfsDiscovered);
  281. dfsWorld.showResults(dfsExitFound, dfsDiscovered, dfsPath);
  282. // 2
  283. cout << endl << "Breadth-First Search" << endl;
  284. World bfsWorld(w);
  285. list<unsigned int> bfsPath;
  286. list<unsigned int> bfsDiscovered;
  287. bool bfsExitFound = bfsWorld.bfs(start, end, bfsPath, bfsDiscovered);
  288. bfsWorld.showResults(bfsExitFound, bfsDiscovered, bfsPath);
  289. // End
  290. return 0;
  291. }