Depth First Search Algorithm
The Depth First Search (DFS) algorithm is a widely used graph traversal technique that explores nodes in a graph as deep as possible before backtracking. It operates on the principle of visiting all the vertices of a graph in a depthward fashion, traversing from the starting node to the farthest node along a particular path and then backtracking to explore other paths. DFS can be implemented using recursion or an explicit stack data structure, and it is commonly used for solving problems such as pathfinding, connectivity, and topological sort in a wide range of applications, including artificial intelligence, network analysis, and game development.
The algorithm begins by selecting a source node and marking it as visited. It then recursively visits each unvisited adjacent node from the current node, marking them as visited as the traversal proceeds. Once there are no more unvisited adjacent nodes, the algorithm backtracks to the previous node in the path and continues searching for unvisited nodes until all the nodes in the graph have been explored. DFS is often contrasted with the Breadth First Search (BFS) algorithm, which explores nodes in a breadthward fashion, visiting all neighbors of a node before proceeding to their neighbors. While both algorithms can be used for similar purposes, DFS is more suitable for problems where the solution lies deep in the search space or when the search space is large and not all nodes need to be explored. 
                      
                      
                      
                        
                      
                      
                    
                   
                  fun main(args: Array<String>) {
    val adjacencyList = AdjacencyList<String>()
    val s = adjacencyList.createVertex("S")
    val a = adjacencyList.createVertex("A")
    val b = adjacencyList.createVertex("B")
    val c = adjacencyList.createVertex("C")
    val d = adjacencyList.createVertex("D")
    val f = adjacencyList.createVertex("F")
    val g = adjacencyList.createVertex("G")
    val e = adjacencyList.createVertex("E")
    adjacencyList.add(Undirected(), s, a)
    adjacencyList.add(Undirected(), a, b)
    adjacencyList.add(Undirected(), a, d)
    adjacencyList.add(Undirected(), a, c)
    adjacencyList.add(Undirected(), b, d)
    adjacencyList.add(Undirected(), d, g)
    adjacencyList.add(Undirected(), d, f)
    adjacencyList.add(Undirected(), f, e)
    print(adjacencyList)
    print(depthFirstSearch(s, e, adjacencyList))
}
fun depthFirstSearch(start: VertexString, end: VertexString, graph: AdjacencyList<String>): Stack<VertexString> {
    val visited: MutableSet<VertexString> = mutableSetOf()
    val stack = Stack<VertexString>()
    stack.push(start)
    visited.add(start)
    var vertex = stack.peek()
    loop@while (vertex != null && vertex != end) {
        val neighbors = graph.edges(vertex)
        if (neighbors != null && neighbors.count() > 0) {
            for (edge in neighbors) {
                if (!visited.contains(edge.destination)) {
                    visited.add(edge.destination)
                    stack.push(edge.destination)
                    print("$stack")
                    vertex = stack.peek()
                    continue@loop
                }
            }
        } else {
            print("backtrack from $vertex")
            stack.pop()
            vertex = stack.peek()
            continue
        }
        print("backtrack from $vertex")
        stack.pop()
        vertex = stack.peek()
    }
    return stack
}