kotlin coding test course, representation of graphs

Hello! Today, we will take a detailed look at how to represent graphs, which frequently appear in coding tests, using Kotlin. Graphs are an important data structure in various fields and have essential fundamental functions for solving a range of problems. In this article, we will explain how to represent a graph and how to use that representation to solve problems.

What is a Graph?

A graph is a data structure composed of nodes (vertices) and edges. A node represents an object, and an edge represents the relationship between nodes. Graphs can be classified into directed and undirected graphs based on directionality, and into weighted and unweighted graphs based on the presence of weights.

Ways to Represent a Graph

Graphs can be represented in various ways, with the two most common methods being the Adjacency Matrix and the Adjacency List.

1. Adjacency Matrix

An adjacency matrix is a method of representing a graph using a 2D array of size N x N. N is the number of nodes in the graph, and each element of the matrix indicates whether nodes are connected. For example, if node A is connected to node B, then matrix[A][B] will be 1.

    fun createAdjacencyMatrix(vertices: Int, edges: List>): Array> {
        val matrix = Array(vertices) { Array(vertices) { 0 } }
        for (edge in edges) {
            val (from, to) = edge
            matrix[from][to] = 1
            matrix[to][from] = 1  // For undirected graphs
        }
        return matrix
    }

2. Adjacency List

An adjacency list is a method that uses a list to store connected nodes for each node. This method is memory efficient and advantageous for graphs with fewer nodes or edges.

    fun createAdjacencyList(vertices: Int, edges: List>): List> {
        val list = MutableList(vertices) { mutableListOf() }
        for (edge in edges) {
            val (from, to) = edge
            list[from].add(to)
            list[to].add(from)  // For undirected graphs
        }
        return list
    }

Problem: Friend Relationship Network

The following is a problem that represents a graph of friendships.

Problem: Given N people and their friendships, write a program to check whether a friendship exists between two people. Friend relationships are represented as undirected graphs. The numbers of the two people are given sequentially from 0 to N-1.

Problem-Solving Process

1. Understand the Input Format

First, we need to understand the given data. The input is as follows:

  • The first line contains the number of people N.
  • The second line contains the number of friendships M.
  • Subsequently, M lines contain two integers a and b representing each friendship.

2. Choose a Data Structure

To store the given friendships, we will use an adjacency list. This allows for easy exploration of friendships.

3. Create the Graph

    fun main() {
        val reader = BufferedReader(InputStreamReader(System.`in`))
        val n = reader.readLine().toInt()
        val m = reader.readLine().toInt()

        val edges = mutableListOf>()
        for (i in 0 until m) {
            val (a, b) = reader.readLine().split(" ").map { it.toInt() }
            edges.add(Pair(a, b))
        }

        val adjacencyList = createAdjacencyList(n, edges)
    }

4. Check Friendship

The method to check whether two specific people are friends is to use DFS (Depth First Search) or BFS (Breadth First Search) to verify connectivity. Here, we will implement it using DFS.

    fun isConnected(adjList: List>, start: Int, target: Int, visited: BooleanArray): Boolean {
        if (start == target) return true
        visited[start] = true
        
        for (neighbor in adjList[start]) {
            if (!visited[neighbor]) {
                if (isConnected(adjList, neighbor, target, visited)) {
                    return true
                }
            }
        }
        return false
    }

    fun main() {
        // Continuing from the previous code
        val query = reader.readLine().split(" ").map { it.toInt() }
        val (x, y) = query

        val visited = BooleanArray(n) { false }
        val result = isConnected(adjacencyList, x, y, visited)
        println(if (result) "YES" else "NO")
    }

Conclusion

In this lesson, we learned about the basic concepts and methods of representing graphs, and how to solve the friend relationship network problem using these representations. Graphs are essential for solving various problems, and these graph representation and traversal techniques are very useful in algorithm tests.

Having implemented examples using Kotlin, I encourage you to practice by solving various graph problems yourself. Next time, we will discuss the differences between BFS and DFS, as well as various problem-solving strategies utilizing them. I hope this aids you in your learning journey!

Author: [Your Name]

Date: [Date]