-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmissionariesandcannibals.java
126 lines (113 loc) · 4.81 KB
/
missionariesandcannibals.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.Queue;
public class missionariesandcannibals {
HashMap<ArrayList<Integer>, HashSet<ArrayList<Integer>>> map = new HashMap<ArrayList<Integer>, HashSet<ArrayList<Integer>>>();
Queue<ArrayList<Integer>> searchSpace = new LinkedList<ArrayList<Integer>>();
public int MISSIONARIES;
public int CANNIBALS;
/**
* Takes in an input of missionaries and cannibals, and tells the user all the possible states, and all the states connected to those possible states
* The output is formatted as so
* [state]: (possible connecting state 1) (possible connecting state 2) ect...
* The program also says if there's a possible end state where all the people reach the other end of the river
* @param missioanries the number of starting missionaries
* @param cannibals the number of starting cannibals
*/
public missionariesandcannibals(int missioanries, int cannibals) {
MISSIONARIES = missioanries;
CANNIBALS = cannibals;
ArrayList<Integer> initalState = new ArrayList<Integer>();
initalState.add(MISSIONARIES);
initalState.add(CANNIBALS);
initalState.add(1);
searchSpace.add(initalState);
map.put(initalState, new HashSet<ArrayList<Integer>>());
int nextSide = -1;
while(!searchSpace.isEmpty()) {
ArrayList<Integer> currentState = searchSpace.poll();
if(currentState.get(2) == 0)
nextSide = 1;
else
nextSide = 0;
for (int i = 0; i <= MISSIONARIES; i++) {
for (int j = 0; j <= CANNIBALS; j++) {
if(possibleState(currentState,i,j) && validState(i,j) && !(i == currentState.get(0) && j == currentState.get(1))){
ArrayList<Integer> newSpace = new ArrayList<Integer>();
newSpace.add(i);
newSpace.add(j);
newSpace.add(nextSide);
// System.out.print("Adding: ");
// for (int num : newSpace){
// System.out.print(num + " ");
// }
// System.out.print(" for previous state:");
// for (int num : currentState){
// System.out.print(num + " ");
// }
// System.out.println();
if(map.get(newSpace) == null) {
// System.out.println("state does not exist, adding");
HashSet<ArrayList<Integer>> toAdd = new HashSet<ArrayList<Integer>>();
toAdd.add(currentState);
map.put(newSpace, toAdd);
searchSpace.add(newSpace);
} else {
// System.out.println("Found existing");
map.get(newSpace).add(currentState);
}
}
}
}
}
for (ArrayList<Integer> key : map.keySet()) {
System.out.print("[ ");
for (int num : key){
System.out.print(num + " ");
}
System.out.print("]: ");
for (ArrayList<Integer> value : map.get(key)){
System.out.print("(");
for (int value_num : value){
System.out.print(value_num + " ");
}
System.out.print(") ");
}
System.out.println();
}
ArrayList<Integer> testGoal = new ArrayList<Integer>();
testGoal.add(0);
testGoal.add(0);
testGoal.add(0);
if (map.get(testGoal) != null) {
System.out.println("Has possible goal state");
} else
System.out.println("No possible goal state");
}
/**
* Returns if this state is possible to get to from the current state(that is, only less than 2 difference from the current state)
*/
private boolean possibleState(ArrayList<Integer> currentState, int i, int j) {
if (Math.abs(currentState.get(0) - i) + Math.abs(currentState.get(1) - j) <= 2)
if (currentState.get(2) == 1) {
return (i <= currentState.get(0) && j <= currentState.get(1));
} else { //the side is 0
return (i >= currentState.get(0) && j >= currentState.get(1));
}
return false;
}
/**
* Returns true if this state is a valid state(that is, missionaries will not get eaten)
*/
private boolean validState(int i, int j) {
if ((i >= j || i == 0) && ((MISSIONARIES - i >= CANNIBALS - j) || (MISSIONARIES - i == 0))) //check if left side is valid AND right side is valid
return true;
else
return false;
}
public static void main(String[] args) {
new missionariesandcannibals(3,3);
}
}