ppeper
by ppeper
2 분 소요

Categories

Tags

최근에 알고리즘의 풀이를 자바언어에서 코틀린언어로 바꾸며 진행해 나가면서 알고리즘 문제 풀이에서 많이 사용되는 정렬을 하는 방법 에 대해서 알아가보려고 한다.


리스트 기본 정렬

코틀린에서 리스트의 정렬은 다음과 같이 크게 sort, sortBy, sortWith의 세가지 방법이 있다.

코틀린에서는 ImmutableMutable 로 나뉜다. 여기서 Immutable(데이터의 변경이 불가능)에서는 함수의 끝에 ed 로 끝난다.

  • Mutable(데이터의 변경이 가능): sort(), reverse()

  • Immutable(데이터의 변경 불가능): sorted(), reversed()

sort(), sorted() / reverse(), reversed()

sort()데이터의 변경이 가능한 리스트에 사용되며, sort()리스트의 원형 변경하지 않고, 리스트를 정렬된(오름차순) 리스트로 변경 합니다.

fun main() {
    val mutablelist = mutableListOf(7, 2, 1, 10, 12, 5, 4)
    mutablelist.sort()
    println("MutableList: $mutablelist")
}

sorted()데이터의 변경이 불가능 한 리스트에 사용되며, sorted()리스트의 원형 변경하지 않고, 정렬된(오름차순) 리스트를 생성하여 반환 합니다.

fun main() {
    val list = listOf(7, 2, 1, 10, 12, 5, 4)
    val sortedlist = list.sorted()
    println("List: $list")
    println("Sortedlist: $sortedlist")
    //    List: [7, 2, 1, 10, 12, 5, 4]
    //    Sortedlist: [1, 2, 4, 5, 7, 10, 12]
}

reverse(), reversed() 또한 위와 같은 방식으로 리스트를 역순으로 정렬 한다.

  • reverse() : Mutable 리스트에 사용한다. 리스트를 역순으로 정렬을 시킨다
  • reversed() : Immutable 리스트에 사용한다. 역순으로 변경된 리스트를 생성하고 반환 한다

기본 정렬 -> sort(ed)By

sortBy()는 리스트 요소가 1개 이상으로 있을때 어떤 요소를 기준으로 정렬을 할지 결정할 수 있다.

아래의 예시에서 Info 클래스에 대해서 이름순으로 정렬을 하고자 하면 sortBy를 사용하면 된다.

data class Info(
    val name: String,
    val count: Int,
    val grade: Double
)

fun main() {
    val list = listOf(
        Info("B", 8, 2.5),
        Info("C", 5, 3.0),
        Info("E", 1, 4.0),
        Info("A", 9, 3.5),
        Info("A", 9, 1.5),
        Info("A", 11, 1.5),
    )

    println("Name으로 정렬")
    list.sortedBy {
        it.name
    }
    //    Name으로 정렬
    //    Info(name=A, count=11, grade=3.5)
    //    Info(name=A, count=9, grade=3.5)
    //    Info(name=A, count=9, grade=1.5)
    //    Info(name=B, count=8, grade=2.5)
    //    Info(name=C, count=5, grade=3.0)
    //    Info(name=E, count=1, grade=4.0)

}

기준이 2개 이상 -> sort(ed)With

위의 예시에서는 Name으로 오름차순으로 정렬을 한것을 볼 수 있다. 여기서 새 프로퍼티 Count를 정렬 조건으로 추가하고 싶다면 sortWith()을 사용하면 된다. sortWith()는 인자로 Comparator를 구현하여 넘겨주게 된다.

fun main() {
    val list = listOf(
        Info("B", 8, 2.5),
        Info("C", 5, 3.0),
        Info("E", 1, 4.0),
        Info("A", 11, 3.5),
        Info("A", 9, 3.5),
        Info("A", 9, 1.5),
    )

    println("Name으로 정렬후 Count으로 정렬")
    list.sortedWith(
        compareBy<Info> {
            it.name
        }.thenBy {
            it.count
        }
    )
    //    Name으로 정렬후 Count으로 정렬
    //    Info(name=A, count=9, grade=3.5)
    //    Info(name=A, count=9, grade=1.5)
    //    Info(name=A, count=11, grade=3.5)
    //    Info(name=B, count=8, grade=2.5)
    //    Info(name=C, count=5, grade=3.0)
    //    Info(name=E, count=1, grade=4.0)
    println("Name으로 정렬후 Count으로 정렬후 Grade로 정렬")
    list.sortedWith(
        compareBy<Info> {
            it.name
        }.thenBy {
            it.count
        }.thenBy {
            it.grade
        }
    )
    //    Name으로 정렬후 Count으로 정렬후 Grade로 정렬
    //    Info(name=A, count=9, grade=1.5)
    //    Info(name=A, count=9, grade=3.5)
    //    Info(name=A, count=11, grade=3.5)
    //    Info(name=B, count=8, grade=2.5)
    //    Info(name=C, count=5, grade=3.0)
    //    Info(name=E, count=1, grade=4.0)
}

정리

정렬의 기본값은 오름차순으로 정렬 이고 이는 decending을 통하여 내림차순으로 정렬 이 가능하다.

  • Immutable(데이터의 변경이 불가능)
    • *ed()를 사용: 정렬된 리스트를 생성하고 반환 한다.
  • Mutable(데이터의 변경이 가능)
    • *()를 사용: 현재 리스트를 정렬한다.
  • 기본정렬 : sort(오름차순), reverse(역순), *Decending(내림차순)
  • sortBy, sortByDecending : 리스트 요소가 1개 이상일때 > 어떤 요소를 기준으로 정렬을 할지 결정할 수 있다.

  • sortWith, sortWithDescending : 기준이 2개이상으로 정렬을 하려고자 할때 사용이 가능하다.
    • compareBy, compareByDescending: Comparator를 사용하여 정렬기준을 정한다.
    • thenBy, thenByDecending: compareBy에 이어서 정렬기준을 정할때 사용한다.