main.cc 9.9 KB

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