Graphs can be in made in 2 ways, Adjacency matrix and Adjacency List. we will be going thru Adjacency List.
I won't go in depth of Adjacency matrix but I want to show the differences between them before we go further
Adjacency Matrix
is made with 2D array and and very costly in space but has faster removal / read times whereas Adjacency List
is space efficient and faster in adding vertices and edges.
There is no telling that one of them is better than the other, it depends on the situation. But here in this blog we will go over Adjacency List.
Lines that connect one node to another node are called edges, and nodes in graphs are called Vertices.
As we walk through in the blog, I want y'all to keep understanding it with the help of the visualizer, there is no use for me to simply every piece when you have the visualizer in front of you 😉.
The plain structure of a graph would look like this:
class Graph {
// Nodes in graphs are unique, there can't be 2 repeating nodes.
// Generally you could use UID's and use same name but I don't want to put this
// code complex
class Node {
public String label;
// Each vertex will have a linked list edges for all the edges it will have
LinkedList<Node> edges = new LinkedList<Node>(); // Linked list that holds all the vertices it points to
public Node(String label) {
this.label = label;
}
@override
public String toString(){
return this.label;
}
}
// use a hashMap here if you want to get the look up time for the nodes in O(1)
// else using linked list will give you a look up time of O(n);
// Since we didn't learn hashmaps yet, I'll be using linked list
private LinkedList<Node> nodes = new LinkedList<Node>();
// Linked list that holds all the vertices (nodes) in the graph
private Node findNode(String label) {}
private Boolean contains(String label) {}
public void addNode(String label) {}
public void removeNode(String label) {}
public void addEdge(String from, String to) {}
public void removeEdge(String from, String to) {}
public void print() {}
}
Before we jump into the methods, since we are using a linked list, we need to make few custom methods to find the node if it exists in the liked list or no.
/*
* Time: O(V) -> V is the number of vertices
* Space: O(1)
*/
private Node findNode(String label) {
// Iterate the linked list, if we have the label return the node
for (Node node : nodes)
if (node.label == label)
return node;
return null;
}
private Boolean contains(String label) {
// Iterate the linked list, if we have the label return true
for (Node node : nodes)
if (node.label == label)
return true;
return false;
}
The First method we will go through is adding vertices
.
Before we go into it, try adding a vertex in the visualizer and see what happens.
So let's look at addNode
method:
You can name it vertex or node what sounds better to you, but in theory people call it vertex.
/*
* Time: O(v) -> where v is the number of vertices
* Space: O(1)
*/
public void addNode(String label) {
// O(v) due to contains method, using hash map would get this to O(1),
// thanks to its fast look up times
Boolean exist = contains(label);
if (exist) // we aren't suppose to add the vertex if it already exist
return;
Node node = new Node(label);
nodes.add(node);
}
Now lets look at addEdge
method, this method makes a path from one vertex to another.
We will go thru removeNode method after this. Try this on the visualizer again 😉.
/*
* Time: O(v)
* Space: O(1)
*/
public void addEdge(String from, String to) {
// We need to first check if both the nodes exist
Node fromNode = findNode(from);
Node toNode = findNode(to);
if (fromNode == null || toNode == null)
return;
fromNode.edges.add(toNode); // This mean we are point from to to.
// If u want to make this undirected graph then add
// toNode.vertices.add(from);
// so we make to also point from 😉
}
Now suppose you want to remove a vertex, then you should also make sure to remove all the links that are connected to the vertex.
So lets look at removeNode
now:
/*
* Time: O(V * E).
* E is the number of edges.
* Space: O(1)
*/
public void removeNode(String label) {
Node labelNode = findNode(label);
if (labelNode == null) // no vertex to remove if it doesn't exist in the linked list
return;
// Using hash map we can get this to O(1), so come back to this after learning
// hash maps 😉
nodes.remove(labelNode);
// We also need to iterate to all the other vertices and remove the link to
// label
// Due to this method, removing a node becomes O(V * E)
for (Node node : nodes)
node.edges.remove(labelNode);
}
Our last method will be removeEdge
:
/*
* Time: O(V + E)
* Space: O(1)
*/
public void removeEdge(String from, String to) {
// Finding node takes O(V)
Node fromNode = findNode(from);
Node toNode = findNode(to);
// we again need to find if both nodes exist or no.
if (fromNode == null || toNode == null)
return;
fromNode.edges.remove(toNode);
}
Just for the purpose of debugging I'll also show a print method:
public void print() {
for (Node node : nodes)
System.out.println(node.label + " is connected to " + node.edges);
}
This Project is Open sourced!!, visit our github to contribute