main.cc 12 KB

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