Skynet, the Bridge

This puzzle is the second level of easy puzzle Skynet, the Chasm - C. This time, there are up to 4 motorbikes on the road, holes everywhere, and you have to lead safely a maximum of motorbikes to the end of the road. As there are only 5 (waiting is useless as you can jump to the same place) actions possible each turn, amongst which 3 (up, down and slow) are rarely needed, and the road is not very long, it is possible to solve this puzzle by exploring the tree of game state. This is what does my solution, until it finds a winning path. Note that the way you explore the tree is determinant : a Depth First Search, avoiding repeating states, discarding useless ones, and favoring most frequent ones (speed, jump) will be needed for your solution to be fast enough.

Java

  1. import java.util.*;
  2. import java.io.*;
  3. import java.math.*;
  4.  
  5. /** Class representing a motorbike. A motorbike has 3 properties :
  6. * <ul>
  7. * <li> Its X and Y coordinates on the road
  8. * <li> A boolean A true if and only if the motorbike is activated
  9. * </ul>
  10. */
  11. class Moto {
  12. int X;
  13. int Y;
  14. boolean A;
  15. public Moto(int X, int Y, boolean A){
  16. this.X = X;
  17. this.Y = Y;
  18. this.A = A;
  19. }
  20. public Moto(Moto m){
  21. this.X = m.X;
  22. this.Y = m.Y;
  23. this.A = m.A;
  24. }
  25. }
  26.  
  27. /** This class represents a step of the run, ie the positions of the
  28. * motorbikes, and their speed
  29. */
  30. class State {
  31. Moto[] motos;
  32. int s;
  33. public State(Moto[] motos, int s) {
  34. this.motos = motos;
  35. this.s = s;
  36. }
  37. public State(State s) {
  38. this.motos = new Moto[s.motos.length];
  39. for (int i = 0; i < s.motos.length; i++)
  40. this.motos[i] = new Moto(s.motos[i]);
  41. this.s = s.s;
  42. }
  43. }
  44.  
  45. class Player {
  46.  
  47. /* Returns true if there is a hole between the moto m
  48. * and its next position on the lane
  49. */
  50. static public boolean willFall (Moto m, int s, String lane){
  51. // The area to examine go from the moto to
  52. // either the end of the road, or the next position
  53. for (int x = m.X+1; x <= Math.min(m.X+s, lane.length()-1); x++)
  54. if (lane.charAt(x) == '0')
  55. return true;
  56. return false;
  57. }
  58. /* Put the moto on its next position
  59. * or disactivates it if it is going to fall in a hole
  60. */
  61. static public void move (Moto m, int s, String lane) {
  62. if (willFall(m, s, lane))
  63. m.A = false;
  64. else
  65. m.X += s;
  66. }
  67.  
  68. /** This inner abstract class represents an action (slow, speed, etc).
  69. * the name should be the string to print to perform this action.
  70. * the function 'apply' applies the action to a state and returns the new state.
  71. * it should also returns null if the action can't be performed, or will lead
  72. * to a useless, or already encoutered state, for performance purposes
  73. */
  74. static abstract class Action {
  75. protected String name;
  76. /** Applies the action to state 's' and returns the new state*/
  77. abstract public State apply(State s);
  78. }
  79.  
  80. static class SPEED extends Action {
  81. public SPEED(){
  82. this.name = "SPEED";
  83. }
  84.  
  85. public State apply(State s){
  86. State s2 = new State(s);
  87. // increases speed by one, and moves each bike
  88. s2.s++;
  89. for (Moto m : s2.motos)
  90. if (m.A)
  91. move(m, s2.s, lanes[m.Y]);
  92. return s2;
  93. }
  94. }
  95. static class SLOW extends Action {
  96.  
  97. public SLOW(){
  98. this.name = "SLOW";
  99. }
  100.  
  101. public State apply(State s){
  102. if (s.s < 2)
  103. return null;
  104. else {
  105. State s2 = new State(s);
  106. // decreases speed by one and moves each bike
  107. s2.s--;
  108. for (Moto m : s2.motos)
  109. if (m.A)
  110. move(m, s2.s, lanes[m.Y]);
  111. return s2;
  112. }
  113. }
  114. }
  115.  
  116. static class UP extends Action {
  117. public UP(){
  118. this.name = "UP";
  119. }
  120.  
  121. public State apply(State s){
  122. if (s.s == 0)
  123. return null;
  124.  
  125. // if moving UP is not possible
  126. for (Moto m : s.motos)
  127. if (m.A && m.Y == 0)
  128. return null;
  129. State s2 = new State(s);
  130. // for each bike, if there is a hole on the concerned area of the current lane
  131. // disactives the bike, else, moves it up, and moves it forward
  132. for (Moto m : s2.motos)
  133. if (m.A)
  134. if (willFall(m, s2.s-1, lanes[m.Y]))
  135. m.A = false;
  136. else {
  137. m.Y--;
  138. move(m, s2.s, lanes[m.Y]);
  139. }
  140. return s2;
  141. }
  142. }
  143.  
  144. static class DOWN extends Action {
  145.  
  146. public DOWN(){
  147. this.name = "DOWN";
  148. }
  149. public State apply(State s){
  150. if (s.s == 0)
  151. return null;
  152. // if moving DOWN isnt possible
  153. for (Moto m : s.motos)
  154. if (m.A && (m.Y == 3))
  155. return null;
  156. State s2 = new State(s);
  157. // for each bike, if there is a hole on the concerned area of the current lane
  158. // disactives the bike, else, moves it down, and moves it forward
  159. for (int i = 0 ; i < s2.motos.length ; i++){
  160. Moto m = s2.motos[i];
  161. if (m.A)
  162. if (willFall(m, s2.s-1, lanes[m.Y]))
  163. m.A = false;
  164. else
  165. {
  166. m.Y++;
  167. move(m, s2.s, lanes[m.Y]);
  168. }
  169. }
  170. return s2;
  171. }
  172. }
  173.  
  174. static class JUMP extends Action {
  175. public JUMP(){
  176. this.name = "JUMP";
  177. }
  178. public State apply(State s){
  179. if (s.s == 0)
  180. return null;
  181. State s2 = new State(s);
  182. // for each bike, if there is a hole at the landing spot, disactives it
  183. // else, places it on the landing spot
  184. for (Moto m : s2.motos)
  185. if (m.A)
  186. if (m.X+s2.s < lanes[m.Y].length() && lanes[m.Y].charAt(m.X+s2.s) == '0')
  187. m.A = false;
  188. else
  189. m.X += s2.s;
  190. return s2;
  191. }
  192. }
  193.  
  194. /** array of possible actions */
  195. private static final Action[] instructions = { new SPEED(), new JUMP(),
  196. new UP(), new DOWN(), new SLOW() };
  197.  
  198. /** lanes of the road*/
  199. private static String lanes[] = new String[4];
  200. /** If it is possible to win from state 's', this functions returns the
  201. * string to print to do so. Else, it returns null.
  202. * Winning means leading a minimum amount of 'min' bikes to the end */
  203. public static String winningMove(State s, int min){
  204. /* If the number of active motorbikes is 0, this state is lost, returns null
  205. if the x of an active motorbike is greater than the length of the lane,
  206. this state is won, returns "WAIT" */
  207. int nbOfA = 0;
  208. int x = 0;
  209. for (Moto m : s.motos)
  210. if (m.A) {
  211. nbOfA++;
  212. x = m.X;
  213. }
  214. if (nbOfA < min)
  215. return null;
  216. if (x >= lanes[0].length())
  217. return "WAIT";
  218.  
  219. // tries each possible instruction, calling recursively winningMove
  220. // on all the next states.
  221. // If one of the calls returns a winning instruction, returns it,
  222. // else returns null
  223. for (Action a : instructions) {
  224. State s2 = a.apply(s);
  225. if (s2 != null && winningMove(s2, min) != null)
  226. return a.name;
  227. }
  228. return null;
  229. }
  230.  
  231. public static void main(String args[]) {
  232. Scanner in = new Scanner(System.in);
  233. // amount of motorbikes, and minimum amount of survivors
  234. int M = in.nextInt();
  235. int V = in.nextInt();
  236. in.nextLine();
  237.  
  238. // Initializes lanes of the road
  239. for (int i = 0; i < 4; i++)
  240. lanes[i] = in.nextLine();
  241. int speed;
  242. Moto[] motos = new Moto[M];
  243. State s = null;
  244. while(true){
  245. // creates the current state
  246. speed = in.nextInt();
  247. in.nextLine();
  248. for (int i = 0; i < M; i++) {
  249. motos[i] = new Moto(in.nextInt(), in.nextInt(), (in.nextInt() == 1));
  250. in.nextLine();
  251. }
  252. s = new State(motos, speed);
  253. /* Search for a move saving all the bikes (for the 3rd achievement)
  254. and if it is not possible, for a move saving the minimum amount of bikes*/
  255. String str = winningMove(s, M);
  256. if (str != null)
  257. System.out.println(str);
  258. else
  259. System.out.println(winningMove(s, V));
  260. }
  261. }
  262. }