import java.lang.Math;
import java.util.*;
public class SaveIntegers
{
public static int[][] timeArray;
public static void main
(String[] args
) {
// Test Case 1
//int[][] times = {{0, 1, 1, 1, 1}, {1, 0, 1, 1, 1}, {1, 1, 0, 1, 1}, {1, 1, 1, 0, 1}, {1, 1, 1, 1, 0}};
//int time_limit = 3;
// Test Case 2
//int[][] times = {{0, 2, 2, 2, -1}, {9, 0, 2, 2, -1}, {9, 3, 0, 2, -1}, {9, 3, 2, 0, -1}, {9, 3, 2, 2, 0}};
//int time_limit = 1;
// Infinite Loop Test 1
//int[][] times = {{0, 2, 2, 2, -1}, {9, -1, 2, 2, -1}, {9, 3, 0, 2, -1}, {9, 3, 2, 0, -1}, {9, 3, 2, 2, 0}};
//int time_limit = 3;
// Infinite Loop Test 2
//int[][] times = {{0, 9, 9, 9, 3}, {9, 0, 9, 1, 9}, {1, 9, 0, 9, 9}, {9, 9, -5, 0, 9}, {9, -2, 9, 9, 0}};
//int time_limit = 3;
// lots of looping
int[][] times = {{0, 0, 0, 0, 0}, {1, 0, 9, 9, 9}, {1, 9, 0, 9, 9}, {1, 9, 9, 0, 9}, {0, 9, 9, 9, 0}};
int time_limit = 8;
timeArray = times;
List<Integer> savedBunnies = new ArrayList<Integer>();
//Search for time-gaining infinite loops. Return all if found.
if(findTimeGainingInfiniteLoop())
{
System.
out.
println("Infinite loops!"); //add all bunnies (none at start or bulkhead)
for (int i = 0; i < timeArray.length - 2; i++)
{
savedBunnies.add(i);
}
}
else
{
//TODO: can this get integrated into findBestPath?
//List<List<Integer>> escapePaths = findAllMinLengthPaths(0, timeArray.length - 1, new ArrayList<Integer>());
//if (getDistanceFromPath(escapePaths.get(0)) <= time_limit)
//{
//Can't even save yourself!
//System.out.println("Can't even save yourself!");
//}
List<Integer> bestPath = findBestPath(0, timeArray.length - 1, time_limit, new ArrayList<Integer>());
System.
out.
println("Best path: " + Arrays.
toString(bestPath.
toArray())); savedBunnies = getSavedBunniesFromPath(bestPath);
}
System.
out.
println("Saved bunnies: " + Arrays.
toString(savedBunnies.
toArray())); }
public static Boolean findTimeGainingInfiniteLoop
() {
for (int i = 0; i < timeArray.length; i++)
{
for (int j = 0; j < timeArray[i].length; j++)
{
if (i == j && timeArray[i][j] < 0)
{
//System.out.println("location " + i + " can loop itself for infinite time!");
return true;
}
if (timeArray[i][j] < 0)
{
// Going from location i to location j adds to the clock!
// Can we get from j to i in one less time than is added?
return findPathWithinTime(j, i, (timeArray[i][j] * -1) - 1);
}
}
}
return false;
}
public static Boolean findPathWithinTime
(int start,
int finish,
int time
) {
//System.out.println("Looking for path from " + start + " to " + finish + " below " + time + " excluding " + Arrays.toString(history.toArray()));
if (timeArray[start][finish] <= time)
{
//System.out.println("Found path from " + start + " to " + finish + " under " + time);
return true;
}
if (start == finish)
{
//System.out.println("Find path to self: " + start);
if (time >= 0)
{
return true;
}
else
{
return false;
}
}
List<List<Integer>> minPaths = findAllMinLengthPaths(start, finish, new ArrayList<Integer>());
return getDistanceFromPath(minPaths.get(0)) <= time;
}
public static List<List<Integer>> findAllMinLengthPaths(int start, int finish, List<Integer> history)
{
List<List<Integer>> listOfPaths = new ArrayList<List<Integer>>();
//System.out.println("Finding paths from " + start + " to " + finish + " excluding " + Arrays.toString(history.toArray()));
history.add(start);
List<Integer> directPath = new ArrayList<Integer>(history);
//Might be able to remove this if/check and always add finish to directPath
if (!directPath.contains(finish))
{
directPath.add(finish);
}
listOfPaths.add(directPath);
//for each possible next step, get the min length paths and update listOfPaths
for (int j = 0; j < timeArray[start].length; j++)
{
if (!history.contains(j) && j != finish)
{
List<List<Integer>> nextPaths = findAllMinLengthPaths(j, finish, new ArrayList<Integer>(history));
int nextPathLength = getDistanceFromPath(nextPaths.get(0));
int currentPathLength = getDistanceFromPath(listOfPaths.get(0));
if (currentPathLength > nextPathLength)
{
//System.out.println("Shorter path from " + start + " to " + finish + ": " + Arrays.toString(nextPaths.get(0).toArray()));
listOfPaths.clear();
listOfPaths.addAll(nextPaths);
}
if (currentPathLength == nextPathLength)
{
listOfPaths.addAll(nextPaths);
}
}
}
if (listOfPaths.size() > 0)
{
//System.out.println("Found paths from " + start + " to " + finish + " including " + Arrays.toString(listOfPaths.get(0).toArray()));
}
else
{
System.
out.
println("Found no paths from " + start
+ " to " + finish
+ " shorter than " + Arrays.
toString(history.
toArray())); }
return listOfPaths;
}
public static List<Integer> findBestPath(int start, int finish, int time_limit, List<Integer> history)
{
List
<Integer
> targetList
= new ArrayList
<Integer
>(Arrays.
asList(0,
1,
0,
2,
0,
3,
0)); history.add(start);
if (history.equals(targetList))
{
System.
out.
println("Found solution: " + Arrays.
toString(history.
toArray())); }
List<Integer> workingPath = new ArrayList<Integer>(history);
if (workingPath.get(workingPath.size() - 1) != finish)
{
workingPath.add(finish);
}
//for each possible next step, get the best paths and update workingPath
for (int j = 0; j < timeArray[start].length; j++)
{
if (validMove(j, history))
{
if (history.equals(targetList))
{
System.
out.
println("Calling findBestPath for " + Arrays.
toString(history.
toArray()) + " to " + j
); }
List<Integer> nextPath = findBestPath(j, finish, time_limit, new ArrayList<Integer>(history));
targetList.add(4);
if (nextPath.equals(targetList))
{
System.
out.
println("Now looking at answer " + Arrays.
toString(nextPath.
toArray()) + ", j: " + j
+ " history: " + Arrays.
toString(history.
toArray())); }
int nextPathLength = getDistanceFromPath(nextPath);
if (nextPath.size() > history.size() + 2)
{
System.
out.
println("Size of " + Arrays.
toString(nextPath.
toArray()) + " is > 2 more than " + Arrays.
toString(history.
toArray())); }
if (time_limit >= nextPathLength)
{
if (nextPath.equals(targetList))
{
System.
out.
println("Answer valid for " + Arrays.
toString(history.
toArray()) + " with j: " + j
); }
//System.out.println("Length " + nextPathLength + " of " + Arrays.toString(nextPath.toArray()) + " is under " + time_limit);
List<Integer> currentPathSaved = getSavedBunniesFromPath(workingPath);
List<Integer> nextPathSaved = getSavedBunniesFromPath(nextPath);
if (isBetterListOfSavedBunnies(nextPathSaved, currentPathSaved))
{
System.
out.
println("Replacing path " + Arrays.
toString(workingPath.
toArray()) + " with " + Arrays.
toString(nextPath.
toArray())); workingPath = nextPath;
}
//else
//{
// System.out.println("List " + Arrays.toString(nextPathSaved.toArray()) + " isn't better than " + Arrays.toString(currentPathSaved.toArray()));
//}
}
//else
//{
// System.out.println("Length " + nextPathLength + " of " + Arrays.toString(nextPath.toArray()) + " is over " + time_limit);
//}
}
//else
//{
// System.out.println("Invalid move " + j + " on " + Arrays.toString(history.toArray()));
//}
//if (j == timeArray[start].length - 1)
//{
// System.out.println("Finishing " + Arrays.toString(history.toArray()) + " and workingPath is " + Arrays.toString(workingPath.toArray()));
//}
}
if (history.equals(targetList))
{
System.
out.
println("Returning answer: " + Arrays.
toString(workingPath.
toArray())); }
return workingPath;
}
public static Boolean validMove
(int move, List
<Integer
> history
) {
//We've already found infinite loops by standing still. Don't allow others.
if (move == history.get(history.size() - 1))
{
return false;
}
//Moving to the bulkhead or start are almost always allowed
if (move == timeArray.length - 1 || move == 0)
{
//Don't allow start-bulkhead infinite loops
if (history.size() >= 2 &&
history.get(history.size() - 1) == 0 &&
history.get(history.size() - 2) == timeArray.length - 1)
{
//System.out.println("Rejecting loop: " + Arrays.toString(history.toArray()) + ", move " + move);
return false;
}
return true;
}
//Otherwise it's only valid if we haven't already grabbed the bunny
return !history.contains(move);
}
public static int getDistanceFromPath(List<Integer> history)
{
//System.out.print("Distance of " + Arrays.toString(history.toArray()) + " is ");
if (history.size() <= 1)
{
//System.out.print(0);
//System.out.println();
return 0;
}
//int historyHash = history.hashCode();
//if (hmap.containsKey(historyHash))
//{
// System.out.print("hashed as " + hmap.get(historyHash));
// System.out.println();
// return hmap.get(historyHash);
//}
Iterator<Integer> distanceIter = history.iterator();
int distance = 0, start = distanceIter.next();
while (distanceIter.hasNext())
{
int next = distanceIter.next();
distance += timeArray[start][next];
start = next;
}
//System.out.print(distance);
// System.out.println();
//hmap.put(historyHash, distance);
return distance;
}
public static List<Integer> getSavedBunniesFromPath(List<Integer> history)
{
List<Integer> savedBunnies = new ArrayList<Integer>();
Iterator<Integer> saveIter = history.iterator();
while (saveIter.hasNext())
{
int move = saveIter.next();
if (move != 0 && move != timeArray.length - 1)
{
savedBunnies.add(move-1);
}
}
//System.out.println("Saved " + Arrays.toString(savedBunnies.toArray()) + " from " + Arrays.toString(history.toArray()));
return savedBunnies;
}
public static Boolean isBetterListOfSavedBunnies
(List
<Integer
> firstBunnyList, List
<Integer
> secondBunnyList
) {
if (firstBunnyList.size() == secondBunnyList.size())
{
for (int i = 0; i < firstBunnyList.size(); i++)
{
if (firstBunnyList.get(i) < secondBunnyList.get(i))
{
System.
out.
println("Better " + Arrays.
toString(firstBunnyList.
toArray()) + " than " + Arrays.
toString(secondBunnyList.
toArray())); return true;
}
else if (firstBunnyList.get(i) > secondBunnyList.get(i))
{
System.
out.
println("Worse " + Arrays.
toString(firstBunnyList.
toArray()) + " than " + Arrays.
toString(secondBunnyList.
toArray())); return false;
}
}
}
else
{
if (firstBunnyList.size() > secondBunnyList.size())
{
System.
out.
println("Better size " + Arrays.
toString(firstBunnyList.
toArray()) + " than " + Arrays.
toString(secondBunnyList.
toArray())); }
else
{
System.
out.
println("Worse size " + Arrays.
toString(firstBunnyList.
toArray()) + " than " + Arrays.
toString(secondBunnyList.
toArray())); }
return firstBunnyList.size() > secondBunnyList.size();
}
//System.out.println("Equal " + Arrays.toString(firstBunnyList.toArray()) + " and " + Arrays.toString(secondBunnyList.toArray()));
// equal lists
return false;
}
}
import java.lang.Math;
import java.util.*;

public class SaveIntegers
{
  public static int[][] timeArray;
  
  public static void main(String[] args)
  {
    // Test Case 1
    //int[][] times = {{0, 1, 1, 1, 1}, {1, 0, 1, 1, 1}, {1, 1, 0, 1, 1}, {1, 1, 1, 0, 1}, {1, 1, 1, 1, 0}};
    //int time_limit = 3;
    
    // Test Case 2
    //int[][] times = {{0, 2, 2, 2, -1}, {9, 0, 2, 2, -1}, {9, 3, 0, 2, -1}, {9, 3, 2, 0, -1}, {9, 3, 2, 2, 0}};
    //int time_limit = 1;
    
    // Infinite Loop Test 1
    //int[][] times = {{0, 2, 2, 2, -1}, {9, -1, 2, 2, -1}, {9, 3, 0, 2, -1}, {9, 3, 2, 0, -1}, {9, 3, 2, 2, 0}};
    //int time_limit = 3;
    
    // Infinite Loop Test 2
    //int[][] times = {{0, 9, 9, 9, 3}, {9, 0, 9, 1, 9}, {1, 9, 0, 9, 9}, {9, 9, -5, 0, 9}, {9, -2, 9, 9, 0}};
    //int time_limit = 3;
    
    // lots of looping
    int[][] times = {{0, 0, 0, 0, 0}, {1, 0, 9, 9, 9}, {1, 9, 0, 9, 9}, {1, 9, 9, 0, 9}, {0, 9, 9, 9, 0}};
    int time_limit = 8;
    
    timeArray = times;
    List<Integer> savedBunnies = new ArrayList<Integer>();

    //Search for time-gaining infinite loops. Return all if found.
    if(findTimeGainingInfiniteLoop())
    {
      System.out.println("Infinite loops!");
      //add all bunnies (none at start or bulkhead)
      for (int i = 0; i < timeArray.length - 2; i++)
      {
        savedBunnies.add(i);
      }
    }
    else
    {
      
      //TODO: can this get integrated into findBestPath?
      //List<List<Integer>> escapePaths = findAllMinLengthPaths(0, timeArray.length - 1, new ArrayList<Integer>());
      //if (getDistanceFromPath(escapePaths.get(0)) <= time_limit)
      //{
        //Can't even save yourself!
        //System.out.println("Can't even save yourself!");
      //}

      List<Integer> bestPath = findBestPath(0, timeArray.length - 1, time_limit, new ArrayList<Integer>());
      System.out.println("Best path: " + Arrays.toString(bestPath.toArray()));
      savedBunnies = getSavedBunniesFromPath(bestPath);
    }
    
    System.out.println("Saved bunnies: " + Arrays.toString(savedBunnies.toArray()));
  }
  
  public static Boolean findTimeGainingInfiniteLoop()
  {
    for (int i = 0; i < timeArray.length; i++)
    {
      for (int j = 0; j < timeArray[i].length; j++)
      {
        if (i == j && timeArray[i][j] < 0)
        {
          //System.out.println("location " + i + " can loop itself for infinite time!");
          return true;
        }
        if (timeArray[i][j] < 0)
        {
          // Going from location i to location j adds to the clock!
          // Can we get from j to i in one less time than is added?
          return findPathWithinTime(j, i, (timeArray[i][j] * -1) - 1);
        }
      }
      
    }
    return false;
  }
  
  public static Boolean findPathWithinTime(int start, int finish, int time)
  {
    //System.out.println("Looking for path from " + start + " to " + finish + " below " + time + " excluding " + Arrays.toString(history.toArray()));
    if (timeArray[start][finish] <= time)
    {
      //System.out.println("Found path from " + start + " to " + finish + " under " + time);
      return true;
    }
    if (start == finish)
    {
      //System.out.println("Find path to self: " + start);
      if (time >= 0)
      {
        return true;
      }
      else
      {
        return false;
      }
    }
    List<List<Integer>> minPaths = findAllMinLengthPaths(start, finish, new ArrayList<Integer>());

    return getDistanceFromPath(minPaths.get(0)) <= time;
  }
  
  public static List<List<Integer>> findAllMinLengthPaths(int start, int finish, List<Integer> history)
  {
    List<List<Integer>> listOfPaths = new ArrayList<List<Integer>>();
    //System.out.println("Finding paths from " + start + " to " + finish + " excluding " + Arrays.toString(history.toArray()));
    history.add(start);
    List<Integer> directPath = new ArrayList<Integer>(history);
    //Might be able to remove this if/check and always add finish to directPath
    if (!directPath.contains(finish))
    {
      directPath.add(finish);
    }
    listOfPaths.add(directPath);
    
    //for each possible next step, get the min length paths and update listOfPaths
    for (int j = 0; j < timeArray[start].length; j++)
    {
      if (!history.contains(j) && j != finish)
      {
        List<List<Integer>> nextPaths = findAllMinLengthPaths(j, finish, new ArrayList<Integer>(history));
        int nextPathLength = getDistanceFromPath(nextPaths.get(0));
        int currentPathLength = getDistanceFromPath(listOfPaths.get(0));
        if (currentPathLength > nextPathLength)
        {
          //System.out.println("Shorter path from " + start + " to " + finish + ": " + Arrays.toString(nextPaths.get(0).toArray()));
          listOfPaths.clear();
          listOfPaths.addAll(nextPaths);
        }
        if (currentPathLength == nextPathLength)
        {
          listOfPaths.addAll(nextPaths);
        }
      }           
    }

    if (listOfPaths.size() > 0)
    {
		//System.out.println("Found paths from " + start + " to " + finish + " including " + Arrays.toString(listOfPaths.get(0).toArray()));
    }
    else
    {
      System.out.println("Found no paths from " + start + " to " + finish + " shorter than " + Arrays.toString(history.toArray()));
    }
    return listOfPaths;
  }
  
  public static List<Integer> findBestPath(int start, int finish, int time_limit, List<Integer> history)
  {
    List<Integer> targetList = new ArrayList<Integer>(Arrays.asList(0, 1, 0, 2, 0, 3, 0));
    history.add(start);
    if (history.equals(targetList))
    {
      System.out.println("Found solution: " + Arrays.toString(history.toArray()));
    }
    List<Integer> workingPath = new ArrayList<Integer>(history);
    if (workingPath.get(workingPath.size() - 1) != finish)
    {
      workingPath.add(finish);
    }
    //for each possible next step, get the best paths and update workingPath
    for (int j = 0; j < timeArray[start].length; j++)
    {
      if (validMove(j, history))
      {
        if (history.equals(targetList))
        {
        	System.out.println("Calling findBestPath for " + Arrays.toString(history.toArray()) + " to " + j);
        }
        List<Integer> nextPath = findBestPath(j, finish, time_limit, new ArrayList<Integer>(history));
        targetList.add(4);
        if (nextPath.equals(targetList))
        {
        	System.out.println("Now looking at answer " + Arrays.toString(nextPath.toArray()) + ", j: " + j + " history: " + Arrays.toString(history.toArray()));
        }
        int nextPathLength = getDistanceFromPath(nextPath);
        if (nextPath.size() > history.size() + 2)
        {
          System.out.println("Size of " + Arrays.toString(nextPath.toArray()) + " is > 2 more than " + Arrays.toString(history.toArray()));
        }
        if (time_limit >= nextPathLength)
        {
          if (nextPath.equals(targetList))
          {
            System.out.println("Answer valid for " + Arrays.toString(history.toArray()) + " with j: " + j);
          }
          //System.out.println("Length " + nextPathLength + " of " + Arrays.toString(nextPath.toArray()) + " is under " + time_limit);
          List<Integer> currentPathSaved = getSavedBunniesFromPath(workingPath);
          List<Integer> nextPathSaved = getSavedBunniesFromPath(nextPath);
          if (isBetterListOfSavedBunnies(nextPathSaved, currentPathSaved))
          {
            System.out.println("Replacing path " + Arrays.toString(workingPath.toArray()) + " with " + Arrays.toString(nextPath.toArray()));
            workingPath = nextPath;
          }
          //else
          //{
          //  System.out.println("List " + Arrays.toString(nextPathSaved.toArray()) + " isn't better than " + Arrays.toString(currentPathSaved.toArray()));
          //}
        }
        //else
        //{
        //  System.out.println("Length " + nextPathLength + " of " + Arrays.toString(nextPath.toArray()) + " is over " + time_limit);
        //}
      }
      //else
      //{
      //  System.out.println("Invalid move " + j + " on " + Arrays.toString(history.toArray()));
      //}
      //if (j == timeArray[start].length - 1)
      //{
      //  System.out.println("Finishing " + Arrays.toString(history.toArray()) + " and workingPath is " + Arrays.toString(workingPath.toArray()));
      //}
    }
    if (history.equals(targetList))
    {
      System.out.println("Returning answer: " + Arrays.toString(workingPath.toArray()));
    }

    return workingPath;        
  }
  
  public static Boolean validMove(int move, List<Integer> history)
  {
    //We've already found infinite loops by standing still. Don't allow others.
    if (move == history.get(history.size() - 1))
    {
      return false;
    }
    //Moving to the bulkhead or start are almost always allowed
    if (move == timeArray.length - 1 || move == 0)
    {
      //Don't allow start-bulkhead infinite loops
      if (history.size() >= 2 &&
        history.get(history.size() - 1) == 0 &&
        history.get(history.size() - 2) == timeArray.length - 1)
      {
        //System.out.println("Rejecting loop: " + Arrays.toString(history.toArray()) + ", move " + move);
        return false;
      }
      return true;
    }
    //Otherwise it's only valid if we haven't already grabbed the bunny
    return !history.contains(move);
  }
  
  public static int getDistanceFromPath(List<Integer> history)
  {
    //System.out.print("Distance of " + Arrays.toString(history.toArray()) + " is ");
    if (history.size() <= 1)
    {
      //System.out.print(0);
      //System.out.println();
      return 0;
    }
    //int historyHash = history.hashCode();
    //if (hmap.containsKey(historyHash))
    //{
    //  System.out.print("hashed as " + hmap.get(historyHash));
    //        System.out.println();
    //  return hmap.get(historyHash);
    //}
    Iterator<Integer> distanceIter = history.iterator();
    int distance = 0, start = distanceIter.next();
    while (distanceIter.hasNext())
    {
      int next = distanceIter.next();
      distance += timeArray[start][next];
      start = next;
    }
    //System.out.print(distance);
    //      System.out.println();
    //hmap.put(historyHash, distance);
    return distance;
  }
  
  public static List<Integer> getSavedBunniesFromPath(List<Integer> history)
  {
    List<Integer> savedBunnies = new ArrayList<Integer>();
    Iterator<Integer> saveIter = history.iterator();
    while (saveIter.hasNext())
    {
      int move = saveIter.next();
      if (move != 0 && move != timeArray.length - 1)
      {
        savedBunnies.add(move-1);
      }
    }
    //System.out.println("Saved " + Arrays.toString(savedBunnies.toArray()) + " from " + Arrays.toString(history.toArray()));
    return savedBunnies;
  }

  public static Boolean isBetterListOfSavedBunnies(List<Integer> firstBunnyList, List<Integer> secondBunnyList)
  {
    if (firstBunnyList.size() == secondBunnyList.size())
    {
      for (int i = 0; i < firstBunnyList.size(); i++)
      {
        if (firstBunnyList.get(i) < secondBunnyList.get(i))
        {
          System.out.println("Better " + Arrays.toString(firstBunnyList.toArray()) + " than " + Arrays.toString(secondBunnyList.toArray()));
          return true;
        }
        else if (firstBunnyList.get(i) > secondBunnyList.get(i))
        {
          System.out.println("Worse " + Arrays.toString(firstBunnyList.toArray()) + " than " + Arrays.toString(secondBunnyList.toArray()));
          return false;
        }
      }
    }
    else
    {
      if (firstBunnyList.size() > secondBunnyList.size())
      {
        System.out.println("Better size " + Arrays.toString(firstBunnyList.toArray()) + " than " + Arrays.toString(secondBunnyList.toArray()));
      }
      else
      {
        System.out.println("Worse size " + Arrays.toString(firstBunnyList.toArray()) + " than " + Arrays.toString(secondBunnyList.toArray()));
      }
      return firstBunnyList.size() > secondBunnyList.size();
    }
    //System.out.println("Equal " + Arrays.toString(firstBunnyList.toArray()) + " and " + Arrays.toString(secondBunnyList.toArray()));
    // equal lists
    return false;
  }
}