fork download
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <stdbool.h>
  4. #include <string.h>
  5. #include <stdint.h>
  6.  
  7. typedef uint32_t uint;
  8.  
  9.  
  10. #define ROWS 4
  11. #define COLUMNS ROWS
  12. #define SIZE (ROWS * COLUMNS)
  13. #define BLANK_VALUE (SIZE - 1)
  14.  
  15. typedef uint Cost;
  16. typedef uint64_t Board;
  17.  
  18. typedef struct {
  19. Board tiles;
  20. int blank;
  21. } State;
  22.  
  23. typedef struct {
  24. State *ptr;
  25. uint length;
  26. } StateArray;
  27.  
  28. typedef struct {
  29. State *ptr;
  30. uint length, capacity;
  31. } StateVector;
  32.  
  33. typedef struct {
  34. State state;
  35. Cost cost;
  36. } Move;
  37.  
  38. typedef struct {
  39. StateVector solution;
  40. bool solved;
  41. Cost f_limit;
  42. } Solver;
  43.  
  44.  
  45. Cost manhattan[SIZE][SIZE];
  46.  
  47.  
  48. int abs(const int x) {
  49. return x >= 0 ? x : -x;
  50. }
  51.  
  52.  
  53. void StateVector_append(StateVector *self, const State *x) {
  54. if (self->length >= self->capacity) {
  55. if (self->capacity > 0)
  56. self->capacity *= 2;
  57. else
  58. self->capacity = 4;
  59. self->ptr = realloc(self->ptr, self->capacity * sizeof(State));
  60. if (self->ptr == NULL)
  61. exit(1);
  62. }
  63. self->ptr[self->length++] = *x;
  64. }
  65.  
  66. StateArray StateVector_slice(StateVector *self) {
  67. return (StateArray){self->ptr, self->length};
  68. }
  69.  
  70.  
  71. int Board_get(const Board data, const size_t i) {
  72. return (data >> (i * 4)) & 0xFU;
  73. }
  74.  
  75. void Board_set(Board *data, const size_t i, const size_t value) {
  76. (*data) &= ~(0xFULL << (i * 4));
  77. (*data) |= ((Board)value) << (i * 4);
  78. }
  79.  
  80.  
  81. Cost State_h(const State *self) {
  82. Cost cost = 0;
  83. for (int i = 0; i < SIZE; i++)
  84. if (i != self->blank)
  85. cost += manhattan[i][Board_get(self->tiles, i)];
  86. return cost;
  87. }
  88.  
  89.  
  90. bool State_is_goal(const State *self) {
  91. return self->tiles == 18364758544493064720ULL;
  92. }
  93.  
  94.  
  95. bool contains(const StateArray solution, const State *state) {
  96. for (int i = solution.length - 1; i >= 0; i--)
  97. if (solution.ptr[i].tiles == state->tiles)
  98. return true;
  99. return false;
  100. }
  101.  
  102.  
  103. uint State_successors(const State *self, const StateArray solution, Move moves[4]) {
  104. uint moves_len = 0;
  105.  
  106. if ((self->blank + 1) % COLUMNS != 0) {
  107. State s = {self->tiles, self->blank + 1};
  108. Board_set(&s.tiles, self->blank, Board_get(self->tiles, self->blank + 1));
  109. Board_set(&s.tiles, self->blank + 1, BLANK_VALUE);
  110. if (!contains(solution, &s))
  111. moves[moves_len++] = (Move){s, 1U};
  112. }
  113.  
  114. if ((self->blank - COLUMNS) >= 0) {
  115. State s = {self->tiles, self->blank - COLUMNS};
  116. Board_set(&s.tiles, self->blank, Board_get(self->tiles, self->blank - COLUMNS));
  117. Board_set(&s.tiles, self->blank - COLUMNS, BLANK_VALUE);
  118. if (!contains(solution, &s))
  119. moves[moves_len++] = (Move){s, 1U};
  120. }
  121.  
  122. if (self->blank % COLUMNS != 0) {
  123. State s = {self->tiles, self->blank - 1};
  124. Board_set(&s.tiles, self->blank, Board_get(self->tiles, self->blank - 1));
  125. Board_set(&s.tiles, self->blank - 1, BLANK_VALUE);
  126. if (!contains(solution, &s))
  127. moves[moves_len++] = (Move){s, 1U};
  128. }
  129.  
  130. if ((self->blank + COLUMNS) < SIZE) {
  131. State s = {self->tiles, self->blank + COLUMNS};
  132. Board_set(&s.tiles, self->blank, Board_get(self->tiles, self->blank + COLUMNS));
  133. Board_set(&s.tiles, self->blank + COLUMNS, BLANK_VALUE);
  134. if (!contains(solution, &s))
  135. moves[moves_len++] = (Move){s, 1U};
  136. }
  137.  
  138. return moves_len;
  139. }
  140.  
  141.  
  142. void State_show(const State *self) {
  143. const char *horiz = "+----+----+----+----+";
  144. for (int r = 0; r < ROWS; r++) {
  145. puts(horiz);
  146. for (int c = 0; c < COLUMNS; c++) {
  147. const int t = Board_get(self->tiles, r * COLUMNS + c);
  148. if (t == BLANK_VALUE)
  149. printf("| ");
  150. else
  151. printf("| %2d ", t);
  152. }
  153. printf("|\n");
  154. }
  155. puts(horiz);
  156. puts("");
  157. }
  158.  
  159.  
  160. #define Solver_inf UINT32_MAX
  161.  
  162. Cost Solver_dfs(Solver *self, const State *s, const Cost g) {
  163. const Cost f1 = g + State_h(s);
  164. if (f1 > self->f_limit)
  165. return f1;
  166. StateVector_append(&self->solution, s);
  167. if (State_is_goal(s)) {
  168. self->solved = true;
  169. return f1;
  170. }
  171. Cost f_min = Solver_inf;
  172.  
  173. Move moves[4];
  174. const uint moves_len = State_successors(s, StateVector_slice(&self->solution), moves);
  175.  
  176. for (int i = 0; i < moves_len; i++) {
  177. const Cost f2 = Solver_dfs(self, &moves[i].state, g + moves[i].cost);
  178. if (self->solved)
  179. return f2;
  180. if (f2 < f_min)
  181. f_min = f2;
  182. }
  183. self->solution.length--;
  184. return f_min;
  185. }
  186.  
  187. StateArray IDA_star_solve(const State s) {
  188. Solver solver;
  189. solver.solution.ptr = NULL;
  190. solver.solution.length = 0;
  191. solver.solution.capacity = 0;
  192. solver.solved = false;
  193. solver.f_limit = State_h(&s);
  194.  
  195. while (!solver.solved && solver.f_limit < Solver_inf)
  196. solver.f_limit = Solver_dfs(&solver, &s, 0);
  197. return StateVector_slice(&solver.solution);
  198. }
  199.  
  200.  
  201. int main(int argc, char** argv) {
  202. for (int r = 0; r < SIZE; r++)
  203. for (int c = 0; c < SIZE; c++)
  204. manhattan[r][c] = abs(r / COLUMNS - c / COLUMNS) +
  205. abs(r % COLUMNS - c % COLUMNS);
  206.  
  207. const char *input = (argc == 2) ? argv[1] : "0FD1C3B648952A7E";
  208. Board tiles = 0;
  209. for (int i = 0; i < strlen(input); i++) {
  210. const char c = input[i];
  211. if (c >= '0' && c <= '9')
  212. Board_set(&tiles, i, c - '0');
  213. else if (c >= 'A' && c <= 'F')
  214. Board_set(&tiles, i, c - 'A' + 10);
  215. else if (c >= 'a' && c <= 'f')
  216. Board_set(&tiles, i, c - 'a' + 10);
  217. else
  218. exit(1);
  219. }
  220.  
  221. int blank = 0;
  222. for (int i = 0; i < SIZE; i++)
  223. if (Board_get(tiles, i) == BLANK_VALUE) {
  224. blank = i;
  225. break;
  226. }
  227.  
  228. const StateArray path = IDA_star_solve((State){tiles, blank});
  229.  
  230. if (path.length == 0) {
  231. puts("No solutions found.");
  232. } else {
  233. printf("Solution in %u moves.\n\n", path.length - 1);
  234. for (int i = 0; i < path.length; i++)
  235. State_show(&path.ptr[i]);
  236. }
  237.  
  238. return 0;
  239. }
Success #stdin #stdout 1.03s 1856KB
stdin
Standard input is empty
stdout
Solution in 45 moves.

+----+----+----+----+
|  0 |    | 13 |  1 |
+----+----+----+----+
| 12 |  3 | 11 |  6 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |    |  1 |
+----+----+----+----+
| 12 |  3 | 11 |  6 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |    |
+----+----+----+----+
| 12 |  3 | 11 |  6 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
| 12 |  3 | 11 |    |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
| 12 |  3 |    | 11 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
| 12 |    |  3 | 11 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
|    | 12 |  3 | 11 |
+----+----+----+----+
|  4 |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
|  4 | 12 |  3 | 11 |
+----+----+----+----+
|    |  8 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
|  4 | 12 |  3 | 11 |
+----+----+----+----+
|  8 |    |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 | 13 |  1 |  6 |
+----+----+----+----+
|  4 |    |  3 | 11 |
+----+----+----+----+
|  8 | 12 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |    |  1 |  6 |
+----+----+----+----+
|  4 | 13 |  3 | 11 |
+----+----+----+----+
|  8 | 12 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |    |  6 |
+----+----+----+----+
|  4 | 13 |  3 | 11 |
+----+----+----+----+
|  8 | 12 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 | 13 |    | 11 |
+----+----+----+----+
|  8 | 12 |  9 |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 | 13 |  9 | 11 |
+----+----+----+----+
|  8 | 12 |    |  5 |
+----+----+----+----+
|  2 | 10 |  7 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 | 13 |  9 | 11 |
+----+----+----+----+
|  8 | 12 |  7 |  5 |
+----+----+----+----+
|  2 | 10 |    | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 | 13 |  9 | 11 |
+----+----+----+----+
|  8 | 12 |  7 |  5 |
+----+----+----+----+
|  2 |    | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 | 13 |  9 | 11 |
+----+----+----+----+
|  8 |    |  7 |  5 |
+----+----+----+----+
|  2 | 12 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |    |  9 | 11 |
+----+----+----+----+
|  8 | 13 |  7 |  5 |
+----+----+----+----+
|  2 | 12 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|    |  4 |  9 | 11 |
+----+----+----+----+
|  8 | 13 |  7 |  5 |
+----+----+----+----+
|  2 | 12 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  8 |  4 |  9 | 11 |
+----+----+----+----+
|    | 13 |  7 |  5 |
+----+----+----+----+
|  2 | 12 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  8 |  4 |  9 | 11 |
+----+----+----+----+
|  2 | 13 |  7 |  5 |
+----+----+----+----+
|    | 12 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  8 |  4 |  9 | 11 |
+----+----+----+----+
|  2 | 13 |  7 |  5 |
+----+----+----+----+
| 12 |    | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  8 |  4 |  9 | 11 |
+----+----+----+----+
|  2 |    |  7 |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  8 |  4 |  9 | 11 |
+----+----+----+----+
|    |  2 |  7 |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|    |  4 |  9 | 11 |
+----+----+----+----+
|  8 |  2 |  7 |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |    |  9 | 11 |
+----+----+----+----+
|  8 |  2 |  7 |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |    | 11 |
+----+----+----+----+
|  8 |  2 |  7 |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |  7 | 11 |
+----+----+----+----+
|  8 |  2 |    |  5 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |  7 | 11 |
+----+----+----+----+
|  8 |  2 |  5 |    |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |  7 |    |
+----+----+----+----+
|  8 |  2 |  5 | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |    |  7 |
+----+----+----+----+
|  8 |  2 |  5 | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |  5 |  7 |
+----+----+----+----+
|  8 |  2 |    | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  9 |  5 |  7 |
+----+----+----+----+
|  8 |    |  2 | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |    |  5 |  7 |
+----+----+----+----+
|  8 |  9 |  2 | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  5 |    |  7 |
+----+----+----+----+
|  8 |  9 |  2 | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  5 |  2 |  7 |
+----+----+----+----+
|  8 |  9 |    | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  5 |  2 |  7 |
+----+----+----+----+
|  8 |  9 | 11 |    |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |  6 |
+----+----+----+----+
|  4 |  5 |  2 |    |
+----+----+----+----+
|  8 |  9 | 11 |  7 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  3 |    |
+----+----+----+----+
|  4 |  5 |  2 |  6 |
+----+----+----+----+
|  8 |  9 | 11 |  7 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |    |  3 |
+----+----+----+----+
|  4 |  5 |  2 |  6 |
+----+----+----+----+
|  8 |  9 | 11 |  7 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |    |  6 |
+----+----+----+----+
|  8 |  9 | 11 |  7 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |  6 |    |
+----+----+----+----+
|  8 |  9 | 11 |  7 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |  6 |  7 |
+----+----+----+----+
|  8 |  9 | 11 |    |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |  6 |  7 |
+----+----+----+----+
|  8 |  9 |    | 11 |
+----+----+----+----+
| 12 | 13 | 10 | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |  6 |  7 |
+----+----+----+----+
|  8 |  9 | 10 | 11 |
+----+----+----+----+
| 12 | 13 |    | 14 |
+----+----+----+----+

+----+----+----+----+
|  0 |  1 |  2 |  3 |
+----+----+----+----+
|  4 |  5 |  6 |  7 |
+----+----+----+----+
|  8 |  9 | 10 | 11 |
+----+----+----+----+
| 12 | 13 | 14 |    |
+----+----+----+----+