Programming Language/Kotlin

[Kotlin] 인터페이스 깊게 이해하기

ggyongi 2024. 9. 7. 19:43
반응형

실제 자주 사용되는 패턴을 통해 여러 usecase를 익혀보자.

import java.time.Instant

// Task 인터페이스
interface Task {
    val taskId: String
    val taskName: String
    val createdAt: Instant
    val dueAt: Instant
    val assignees: List<String>

    fun startTask()

    // 서브인터페이스: Task의 가변 기능 추가
    interface Mutable {
        fun updateTaskDetails(newName: String, newDueAt: Instant)
    }

    // 서브인터페이스: Task를 상속하며 기능을 제한
    interface ReadOnly : Task {
        override fun startTask() {
            throw UnsupportedOperationException("This task is read-only.")
        }
    }

    // 경량화된 데이터 클래스
    data class SimpleTask(
        override val taskId: String,
        override val taskName: String,
        override val createdAt: Instant,
        override val dueAt: Instant
    ) : ReadOnly { // ReadOnly 인터페이스 구현
        override val assignees: List<String> = emptyList() // 경량화된 상태로 빈 리스트
    }

    // 공통적인 기능을 포함하는 추상 클래스
    abstract class BaseTask(
        createdAt: Instant,
        dueAt: Instant
    ) : Task {
        private var _createdAt: Instant = createdAt
        override val createdAt: Instant
            get() = _createdAt

        private var _dueAt: Instant = dueAt
        override val dueAt: Instant
            get() = _dueAt
    }
}

구조 설명

  • Task 인터페이스: 작업 관리 시스템의 기본 인터페이스로, 작업의 ID, 이름, 생성 시간, 마감 시간, 담당자 목록을 정의합니다. startTask() 메서드를 통해 작업을 시작할 수 있습니다.
  • 서브인터페이스:
    • Mutable: 작업의 세부 정보를 수정할 수 있는 기능을 추가하는 서브인터페이스입니다. updateTaskDetails() 메서드를 통해 작업 이름과 마감 시간을 업데이트할 수 있습니다.  -> 기능 추가 역할!!
    • ReadOnly: 작업이 읽기 전용으로 동작하게 하는 서브인터페이스입니다. startTask() 호출 시 예외를 던져 작업 시작을 제한합니다. -> 기능 제한 역할!!
  • 경량화된 데이터 클래스 (SimpleTask):
    • SimpleTask는 경량화된 읽기 전용 Task로, 작업의 ID, 이름, 생성/마감 시간을 포함하지만 담당자 목록(assignees)은 빈 리스트로 설정되어 있습니다. ReadOnly 인터페이스를 구현하여 작업 시작을 제한합니다.
  • 공통 기능을 제공하는 추상 클래스 (BaseTask):
    • BaseTask는 Task 인터페이스의 공통 속성을 포함하는 추상 클래스입니다. **createdAt**과 dueAt 속성은 읽기 전용이지만, 내부적으로는 가변적(var)으로 처리됩니다. 이를 상속받아 다른 작업 클래스에서 공통 기능을 재사용할 수 있습니다.

 

 

사용예시1: MutableTask 정의

import java.time.Instant

// MutableTask 클래스: Task.Mutable을 구현하여 작업 세부 정보 수정 가능
class MutableTask(
    override val taskId: String,
    override var taskName: String,  // var로 선언하여 수정 가능
    createdAt: Instant,
    dueAt: Instant,
    override var assignees: List<String>  // var로 선언하여 수정 가능
) : Task.BaseTask(createdAt, dueAt), Task.Mutable {

    // Task.Mutable 인터페이스의 메서드 구현: 작업 세부 정보 수정
    override fun updateTaskDetails(newName: String, newDueAt: Instant) {
        this.taskName = newName
        this._dueAt = newDueAt  // BaseTask의 _dueAt 속성에 접근하여 수정
        println("Task details updated: New name = $newName, New due date = $newDueAt")
    }

    // Task 인터페이스의 startTask 메서드 구현
    override fun startTask() {
        println("Task '$taskName' has started.")
    }
}

fun main() {
    // MutableTask 인스턴스 생성
    val mutableTask = MutableTask(
        taskId = "T-2001",
        taskName = "Write API documentation",
        createdAt = Instant.now(),
        dueAt = Instant.now().plusSeconds(86400),  // 1일 후 마감
        assignees = listOf("Alice", "Bob")
    )

    // Task 정보 출력
    println("Task ID: ${mutableTask.taskId}")
    println("Task Name: ${mutableTask.taskName}")
    println("Due At: ${mutableTask.dueAt}")
    println("Assignees: ${mutableTask.assignees}")

    // Task 시작
    mutableTask.startTask()

    // Task 세부 정보 수정
    mutableTask.updateTaskDetails("Update API documentation", Instant.now().plusSeconds(172800)) // 2일 후 마감 변경

    // 수정된 Task 정보 출력
    println("Updated Task Name: ${mutableTask.taskName}")
    println("Updated Due At: ${mutableTask.dueAt}")
}

 

사용예시2: FullTask 클래스 정의

// FullTask 클래스: Task의 구체적인 구현
class FullTask(
    override val taskId: String,
    override val taskName: String,
    createdAt: Instant,
    dueAt: Instant,
    override val assignees: List<String>
) : Task.BaseTask(createdAt, dueAt) {
    
    // Task 인터페이스의 메서드 구현
    override fun startTask() {
        println("Task '$taskName' has started.")
    }
}

fun main() {
    val task = FullTask(
        taskId = "T-1001",
        taskName = "Develop feature X",
        createdAt = Instant.now(),
        dueAt = Instant.now().plusSeconds(604800), // 7일 후 마감
        assignees = listOf("Alice", "Bob")
    )

    // Task 시작
    task.startTask()

    // Task의 속성 출력
    println("Task ID: ${task.taskId}")
    println("Task Name: ${task.taskName}")
    println("Created At: ${task.createdAt}")
    println("Due At: ${task.dueAt}")
    println("Assignees: ${task.assignees}")
}

 

사용예시3: SimpleTask

fun main() {
    // SimpleTask 인스턴스 생성 (읽기 전용)
    val simpleTask = Task.SimpleTask(
        taskId = "T-1002",
        taskName = "Review project proposal",
        createdAt = Instant.now(),
        dueAt = Instant.now().plusSeconds(259200) // 3일 후 마감
    )

    // 읽기 전용 Task 출력
    println("Task ID: ${simpleTask.taskId}")
    println("Task Name: ${simpleTask.taskName}")
    println("Created At: ${simpleTask.createdAt}")
    println("Due At: ${simpleTask.dueAt}")
    println("Assignees: ${simpleTask.assignees}")

    // 읽기 전용 Task 시작 시도 (예외 발생)
    try {
        simpleTask.startTask()
    } catch (e: UnsupportedOperationException) {
        println(e.message) // 출력: This task is read-only.
    }
}