-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathDirectedGraph.java
375 lines (339 loc) · 13.5 KB
/
DirectedGraph.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
import java.util.LinkedList;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Queue;
/**
*A DirectedGraph class
*@author Cameron Zurmuhl
*@version 4/29/2017 3:59 p.m
*Additions Emmett O'Toole 2:48P.M 5-2-17
*/
public class DirectedGraph
{
//////////////////////////////////INNER CLASSES/////////////////////////////////////////////
/**
* Class DirectedGraphNode is an inner class that repersent the verticies in the graph
*/
private static class DirectedGraphNode implements Comparable<DirectedGraphNode>
{
/////////////////////////////FIELDS////////////////////////////////
Facility f ; //the node's facility
boolean visited = false; //boolean for traversal
ArrayList<DirectedGraphEdge> outgoingEdges; //list of outgoing edges
/**
* Constructor for class DirectedGraphNode<K>
* @param facility the facility for the node
*/
public DirectedGraphNode(Facility f)
{
this.f =f;
outgoingEdges = new ArrayList<>();
}
/////////////////////////////METHODS////////////////////////////////
/**
* Method returnClosestNeighbor returns the closest vertex to this vertex
* Closest is in a weighted sense
* @return the closest neighbor
*/
public Facility returnClosestNeighbor()
{
//if the node has no closest neighbor, return null
if(outgoingEdges.isEmpty())
{
return null;
} else { //Sort the outgoing edges list and return the first entry's endingNode
Collections.sort(outgoingEdges);
return outgoingEdges.get(0).endingNode.f;
}
}
/**
* Method returnSortedEdges sorts the node's outgoing edges by weight
* @return a sorted array of edges
*/
public ArrayList<DirectedGraphEdge> returnSortedEdges()
{
Collections.sort(outgoingEdges);
return outgoingEdges;
}
@Override
public int compareTo(DirectedGraphNode n)
{
return this.f.compareTo(n.f);
}
}
/**
* Class DirectedGraphEdge is an inner class that represents the edges in a graph
*/
private static class DirectedGraphEdge implements Comparable<DirectedGraphEdge>
{
//////////////////////////////FIELDS/////////////////////////////////////////
DirectedGraphNode startingNode; //starting node of the edge
DirectedGraphNode endingNode; //ending node of the edge
int weight; //the weight of the graph
/**
* Constructor for class DirectedGraphEdge
* @param startingNode the starting node
* @param endingNode the ending node
* @param w the weight of the edge
*/
public DirectedGraphEdge (DirectedGraphNode startingNode, DirectedGraphNode endingNode, int w)
{
this.startingNode = startingNode;
this.endingNode = endingNode;
weight = w;
//add this edge to startingNode's outgoing edge list
startingNode.outgoingEdges.add(this);
}
@Override
public int compareTo(DirectedGraphEdge de)
{
if(this.weight == de.weight) {return 0;}
else if(this.weight < de.weight ) { return -1;}
else { return 1;}
}
}
///////////////////////////////OUTTER ClASS////////////////////////////////////////////////////
///////////////////////////////////FIELDS//////////////////////////////////////////////////////
private ArrayList<DirectedGraphNode> allNodes; //keeps track of all nodes in the graph
/**
* Constructor for class DirectedGraph
*/
public DirectedGraph()
{
allNodes = new ArrayList<>();
}
///////////////////////////////METHODS//////////////////////////////////////////////////////////////
/**
* Method addNode adds a node to the graph with facility f
* Facility should be unique or node should not be added and method should return false
* @param f the facilityh of the added node
* @return whether insertion was successful
*/
public boolean addNode(Facility f)
{
int i=Collections.binarySearch(allNodes,new DirectedGraphNode(f));
if(i<0){
allNodes.add(-i-1,new DirectedGraphNode(f));
return true;
}
else{
return false;
}
}
/**
* Method addEdge adds an edge from the node with facility one to the node with facility two and weight w.
* Edge should not be added if either nodes do not exist.
* If edge already exists, simply change its weight
* @param one the key of the starting node
* @param two the key of the second node
* @param w the weight of the edge
* @returns true if the edge is successfully added.
*/
public boolean addEdge(Facility one, Facility two, int w )
{
//check if both vertices are in the node's list
DirectedGraphNode startingNode = getVertex(one);
DirectedGraphNode endingNode = getVertex(two);
if(startingNode == null || endingNode == null )
{
return false;
}
//check if there is an existent edge between the vertices. If there is, change the weight
DirectedGraphEdge potentialEdge = getEdge(one, two);
DirectedGraphEdge potentialEdge2= getEdge(two,one);
if(potentialEdge != null || potentialEdge2 != null)
{
potentialEdge.weight = w; //change the weight
potentialEdge2.weight=w;
return true;
}
//last case: no edge between the vertices, create new edge
DirectedGraphEdge newEdge = new DirectedGraphEdge(startingNode, endingNode, w);
DirectedGraphEdge newEdge2 = new DirectedGraphEdge(endingNode, startingNode, w);
return true;
}
/**
* Method getNeighbors returns an arrayList of Facilities containing all the neighbors f can reach in one hop, ordered closest to furthest
* @return the list of neighbors
*/
public ArrayList<Facility> getNeighbors(Facility f)
{
DirectedGraphNode startingNode = getVertex(f); //get the starting vertex
ArrayList<DirectedGraphEdge> neighborEdges = startingNode.returnSortedEdges();
ArrayList<Facility> neighbors = new ArrayList<Facility>();
for(DirectedGraphEdge edge : neighborEdges)
{
neighbors.add(edge.endingNode.f); //add the ending nodes of the edges to neighbors array
}
return neighbors;
}
/**
* Method breadthFirstClosest
* prints out the closest neighbor each node in the graph can reach in one hop, using breadth first order
* @param f the facility of the vertex to start at
*/
public void breadthFirstClosest(Facility f)
{
//set all node instance variables to false so I can call method multiple times
for(DirectedGraphNode node : allNodes)
{
if(node.visited)
{
node.visited = false;
}
}
//obtain the vertex, if applicable
DirectedGraphNode startingVertex = getVertex(f);
if(startingVertex == null ) {return;} //can't traverse
//initialize queue
Queue <DirectedGraphNode> q = new LinkedList<>();
q.add(startingVertex);
startingVertex.visited = true;
//start enqueue/dequeue process
while(!q.isEmpty())
{
DirectedGraphNode u = q.poll();
System.out.println(u.f.toString() + " " + u.returnClosestNeighbor()); //print closest neighbor
//enqueue all neighbors, if not already enqueued
for(DirectedGraphEdge edge: u.outgoingEdges) //for every outgoing edge
{
if(!edge.endingNode.visited) //if the ending node is not already visited
{
edge.endingNode.visited = true;
q.add(edge.endingNode); //enqueue
}
}
}
}
/**
* Method getVertex searches the allNodes ArrayList via binary search for the node with a given key
* @param f the facility of the node to search for
* @return the node with that facility (null if non existent)
*/
public DirectedGraphNode getVertex(Facility f)
{
int i=Collections.binarySearch(allNodes,new DirectedGraphNode(f));
return i<0? null: allNodes.get(i);
}
/**
* Method getFacility searches the allNodes ArrayList via binary search for the node with a given key
* @param f the facility of the node to search for
* @return the facility of a node(null if non existent)
*/
public Facility getFacility(Facility f)
{
int i=Collections.binarySearch(allNodes,new DirectedGraphNode(f));
return i<0? null: allNodes.get(i).f;
}
/**
* Method getEdge returns the edge that connects two vertices, or null if there is none
* @param f1 the key of the starting node
* @param f2 the key of the second node
* @return the edge connecting them
*/
public DirectedGraphEdge getEdge(Facility f1, Facility f2)
{
//check if both vertices exit
DirectedGraphNode startingVertex = getVertex(f1);
DirectedGraphNode endingVertex = getVertex(f2);
if(startingVertex == null || endingVertex == null )
{
return null; //one or more vertex does not exist -- cannot have an edge
}
//Case 2: Find the edge that connects the starting vertex to the ending vertex
for(DirectedGraphEdge edge : startingVertex.outgoingEdges )
{
if(edge.endingNode.equals(endingVertex))
{
return edge;
}
}
return null; //no edge connects them
}
/**
* Method getEdgeWeight returns and edge's weight
* @param edge the edge to examine
* @return its weight
*/
public int getEdgeWeight(Facility one, Facility two)
{
return getEdge(one,two).weight;
}
/**
* Method returnClosestNeighbor returns the closest vertex to the passed paramater
* Closest is in a weighted sense
* @param f the facility of the vertex to examine
* @return the closest neighbor to key
*/
public Facility returnClosestNeighbor(Facility f)
{
DirectedGraphNode node = getVertex(f); //get the node that goes with the key
return node.outgoingEdges.size() == 0 ? null: node.returnClosestNeighbor(); //return null, if applicable, or that node's closes neighbor's key
}
/**
* Get all nodes method
* @return ArrayList allnodes in the graph
*/
public ArrayList<DirectedGraphNode> getNodes(){
return this.allNodes;
}
/**
* Create edges method
* Creates edges between all nodes
*/
public void createEdges(){
//Goes through every node
for(int i=0;i<allNodes.size();i++){
//Goes through every node one after i to the end
for(int j=i+1;j<allNodes.size();j++){
//Adds an edge between these two nodes with weight the distance between their two locations
if(allNodes.get(i).f instanceof Warehouse && allNodes.get(j).f instanceof Warehouse) //do not add edges between warehouses
{
continue;
}
this.addEdge(allNodes.get(i).f,allNodes.get(j).f,allNodes.get(i).f.distanceFrom(allNodes.get(j).f));
}
}
}
/**
* Get closest shop with orders method
* Next shop method
* @param Facility a the facility from
* @param Truck the truck searching for its next shop
* which the closest shop with orders is being found
*/
public Shop nextShop(Facility a,Truck t){
ArrayList<Facility>adj=this.getNeighbors(a); //returns an arraylist of sorted neighbors by edge weight
for(int i=0;i<adj.size();i++){
if(adj.get(i) instanceof Shop){ //if the neighbor is a shop
Shop s = (Shop)adj.get(i);
if(!s.isOrdersEmpty()&&!t.getprevShops().contains(s)){ //if the shop still has orders and hasn't been visited by Truck t
return s; //that's that shop we want
} else {
continue; //continue to next facility
}
}
}
return null; //shops are satisfied
}
/**
* Method nextWarehouse returns the next closest warehouse to a facility's position
* @param a the facility to check for adjacent warehouses
* @return the closest warehouse to a facilty
*/
public Warehouse nextWarehouse(Facility a)
{
ArrayList<Facility>adj=this.getNeighbors(a); //returns an arraylist of sorted neighbors by edge weight
for(int i=0;i<adj.size();i++){
if(adj.get(i) instanceof Warehouse){ //if the neighbor is a shop
Warehouse w = (Warehouse)adj.get(i);
if(w.trucksLeft()){
return w;
} else {
continue;
}
}
}
return null; //no next closest warehouse with available trucks
}
}