main.cc 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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. class World
  14. {
  15. private:
  16. // Number of columns
  17. unsigned int l;
  18. // Number of lines
  19. unsigned int h;
  20. // Size of the array
  21. const unsigned int size;
  22. // Unidimensional array for tiles
  23. int* board;
  24. public:
  25. // Constructor
  26. World(unsigned int l_, unsigned int h_, double p)
  27. :l(l_), h(h_), size(l_ * h_)
  28. {
  29. board = new int[size]();
  30. // Add walls to the first and last columns
  31. for (unsigned int i = 0; i < h; i++)
  32. {
  33. board[i * l] = 1;
  34. board[i * l + l - 1] = 1;
  35. }
  36. // Add walls to the first and last lines
  37. for (unsigned int j = 0; j < l; j++)
  38. {
  39. board[j] = 1;
  40. board[(h - 1) * l + j] = 1;
  41. }
  42. for (unsigned int i = 0; i < h; i++)
  43. {
  44. for (unsigned int j = 0; j < l; j++)
  45. {
  46. // add a wall in this tile with probability p and provided that it is neither
  47. // the starting tile nor the goal tile
  48. if ((double) rand() / RAND_MAX < p && !(i == 1 && j == 1) && !(i == h - 2 && j == l - 2))
  49. {
  50. board[i * l + j] = 1;
  51. }
  52. }
  53. }
  54. }
  55. // Display the world
  56. void display()
  57. {
  58. for (unsigned int i = 0; i < h; i++)
  59. {
  60. for (unsigned int j = 0; j < l; j++)
  61. {
  62. switch (board[i * l + j])
  63. {
  64. case 0:
  65. cout << " ";
  66. break;
  67. case 1:
  68. cout << "#";
  69. break;
  70. }
  71. }
  72. cout << endl;
  73. }
  74. }
  75. // compute the successors of tile number i in world w
  76. // we return the number n of valid successors
  77. // the actual list is in array r where only the first n
  78. // elements are significant
  79. unsigned int successors(unsigned int i, unsigned int r[4])
  80. {
  81. unsigned int n = 0;
  82. if (i >= 0 && i < size && board[i] != 1)
  83. {
  84. // if i is a correct tile number (inside the array and not on a wall)
  85. // look in the four adjacent tiles and keep only those with no wall
  86. const unsigned int moves[] = { i - 1, i + 1, i - l, i + l};
  87. for (unsigned int k = 0; k < 4; k++)
  88. {
  89. if (board[moves[k]] != 1)
  90. {
  91. r[n] = moves[k];
  92. n++;
  93. }
  94. }
  95. }
  96. return n;
  97. }
  98. // Depth-first search
  99. // starting from tile number s0, find a path to tile number t
  100. // return true if such a path exists, false otherwise
  101. // if it exists the path is given in variable path (hence the reference &)
  102. bool dfs(unsigned int s0, unsigned int t, list<unsigned int>& path)
  103. {
  104. bool r = false;
  105. bool explored[size];
  106. stack<unsigned int> open;
  107. open.push(s0);
  108. for (unsigned int k(0); k < size; k ++)
  109. explored[k] = false;
  110. explored[s0] = true;
  111. int current;
  112. int neighbour;
  113. unsigned int succs[4];
  114. unsigned int nbSuccs;
  115. do {
  116. current = open.top();
  117. open.pop();
  118. nbSuccs = successors(current, succs);
  119. for (unsigned int i(0); i < nbSuccs; i++) {
  120. neighbour = succs[i];
  121. if (!explored[neighbour])
  122. open.push(neighbour);
  123. }
  124. // Build path
  125. path.push_back(current);
  126. } while (!r && !open.empty());
  127. return r;
  128. }
  129. };
  130. int main()
  131. {
  132. // Initialise the random number generator
  133. srand(time(0));
  134. // Create a world
  135. World w(20, 10, 0.2);
  136. // Display it
  137. w.display();
  138. // Print the tile numbers of the successors of the starting tile (1, 1)
  139. unsigned int succs[4];
  140. unsigned int n = w.successors(21, succs);
  141. for (unsigned int k = 0; k < n; k++)
  142. {
  143. cout << succs[k] << " ";
  144. }
  145. cout << endl;
  146. return 0;
  147. }