본문으로 건너뛰기

관용구

Kotlin에서 자주 사용되는 임의의 이디엄(idiom) 모음입니다. 선호하는 이디엄이 있다면 pull request를 보내 기여해주세요.

DTO 생성 (POJO/POCO)

data class Customer(val name: String, val email: String)

다음을 제공하는 Customer 클래스를 제공합니다.

  • 모든 속성에 대한 getters (및 var의 경우 setters)
  • equals()
  • hashCode()
  • toString()
  • copy()
  • 모든 속성에 대한 component1(), component2(), ... (참조: Data classes)

함수 파라미터의 기본값

fun foo(a: Int = 0, b: String = "") { ... }

리스트 필터링

val positives = list.filter { x `->` x > 0 }

또는 더 짧게:

val positives = list.filter { it > 0 }

Java와 Kotlin 필터링의 차이점을 알아보세요.

컬렉션에 요소가 있는지 확인

if ("[email protected]" in emailsList) { ... }

if ("[email protected]" !in emailsList) { ... }

문자열 보간

println("Name $name")

Java와 Kotlin 문자열 연결의 차이점을 알아보세요.

표준 입력 안전하게 읽기

// 문자열을 읽고 입력 값을 정수로 변환할 수 없으면 null을 반환합니다. 예: Hi there!
val wrongInt = readln().toIntOrNull()
println(wrongInt)
// null

// 정수로 변환할 수 있는 문자열을 읽고 정수를 반환합니다. 예: 13
val correctInt = readln().toIntOrNull()
println(correctInt)
// 13

자세한 내용은 표준 입력 읽기를 참조하세요.

인스턴스 확인

when (x) {
is Foo `->` ...
is Bar `->` ...
else `->` ...
}

읽기 전용 리스트

val list = listOf("a", "b", "c")

읽기 전용 맵

val map = mapOf("a" to 1, "b" to 2, "c" to 3)

맵 항목 접근

println(map["key"])
map["key"] = value

맵 또는 쌍 리스트 순회

for ((k, v) in map) {
println("$k `->` $v")
}

kvnameage처럼 편리한 이름으로 사용할 수 있습니다.

범위 반복

for (i in 1..100) { ... }  // 닫힌 범위: 100을 포함합니다.
for (i in 1..<100) { ... } // 열린 범위: 100을 포함하지 않습니다.
for (x in 2..10 step 2) { ... }
for (x in 10 downTo 1) { ... }
(1..10).forEach { ... }

Lazy 프로퍼티

val p: String by lazy { // 값은 처음 접근할 때만 계산됩니다.
// 문자열 계산
}

확장 함수

fun String.spaceToCamelCase() { ... }

"Convert this to camelcase".spaceToCamelCase()

싱글톤 생성

object Resource {
val name = "Name"
}

타입 안전한 값에 인라인 value 클래스 사용

@JvmInline
value class EmployeeId(private val id: String)

@JvmInline
value class CustomerId(private val id: String)

EmployeeIdCustomerId를 실수로 혼합하면 컴파일 오류가 발생합니다.

노트

@JvmInline 어노테이션은 JVM 백엔드에서만 필요합니다.

추상 클래스 인스턴스화

abstract class MyAbstractClass {
abstract fun doSomething()
abstract fun sleep()
}

fun main() {
val myObject = object : MyAbstractClass() {
override fun doSomething() {
// ...
}

override fun sleep() { // ...
}
}
myObject.doSomething()
}

If-not-null shorthand

val files = File("Test").listFiles()

println(files?.size) // files가 null이 아니면 size가 출력됩니다.

If-not-null-else shorthand

val files = File("Test").listFiles()

// 간단한 대체 값의 경우:
println(files?.size ?: "empty") // files가 null이면 "empty"가 출력됩니다.

// 코드 블록에서 더 복잡한 대체 값을 계산하려면 `run`을 사용합니다.
val filesSize = files?.size ?: run {
val someSize = getSomeSize()
someSize * 2
}
println(filesSize)

Null일 경우 구문 실행

val values = ...
val email = values["email"] ?: throw IllegalStateException("Email is missing!")

비어 있을 수 있는 컬렉션의 첫 번째 항목 가져오기

val emails = ... // 비어 있을 수 있습니다.
val mainEmail = emails.firstOrNull() ?: ""

Java와 Kotlin 첫 번째 항목 가져오기의 차이점을 알아보세요.

Null이 아닐 경우 실행

val value = ...

value?.let {
... // null이 아니면 이 블록을 실행합니다.
}

Null이 아닐 경우 nullable 값 매핑

val value = ...

val mapped = value?.let { transformValue(it) } ?: defaultValue
// 값 또는 변환 결과가 null이면 defaultValue가 반환됩니다.

When 구문에서 반환

fun transform(color: String): Int {
return when (color) {
"Red" `->` 0
"Green" `->` 1
"Blue" `->` 2
else `->` throw IllegalArgumentException("Invalid color param value")
}
}

try-catch 표현식

fun test() {
val result = try {
count()
} catch (e: ArithmeticException) {
throw IllegalStateException(e)
}

// 결과 작업
}

if 표현식

val y = if (x == 1) {
"one"
} else if (x == 2) {
"two"
} else {
"other"
}

Unit을 반환하는 메서드의 Builder 스타일 사용

fun arrayOfMinusOnes(size: Int): IntArray {
return IntArray(size).apply { fill(-1) }
}

단일 표현식 함수

fun theAnswer() = 42

이는 다음과 같습니다.

fun theAnswer(): Int {
return 42
}

이는 다른 이디엄과 효과적으로 결합되어 코드를 더 짧게 만들 수 있습니다. 예를 들어 when 표현식과 함께:

fun transform(color: String): Int = when (color) {
"Red" `->` 0
"Green" `->` 1
"Blue" `->` 2
else `->` throw IllegalArgumentException("Invalid color param value")
}

객체 인스턴스에서 여러 메서드 호출 (with)

class Turtle {
fun penDown()
fun penUp()
fun turn(degrees: Double)
fun forward(pixels: Double)
}

val myTurtle = Turtle()
with(myTurtle) { //100픽셀 정사각형 그리기
penDown()
for (i in 1..4) {
forward(100.0)
turn(90.0)
}
penUp()
}

객체의 프로퍼티 구성 (apply)

val myRectangle = Rectangle().apply {
length = 4
breadth = 5
color = 0xFAFAFA
}

이는 객체 생성자에 없는 프로퍼티를 구성하는 데 유용합니다.

Java 7의 try-with-resources

val stream = Files.newInputStream(Paths.get("/some/file.txt"))
stream.buffered().reader().use { reader `->`
println(reader.readText())
}

제네릭 타입 정보가 필요한 제네릭 함수

//  public final class Gson {
// ...
// public <T> T fromJson(JsonElement json, Class<T> classOfT) throws JsonSyntaxException {
// ...

inline fun <reified T: Any> Gson.fromJson(json: JsonElement): T = this.fromJson(json, T::class.java)

두 변수 바꾸기

var a = 1
var b = 2
a = b.also { b = a }

코드를 미완성으로 표시 (TODO)

Kotlin의 표준 라이브러리에는 항상 NotImplementedError를 throw하는 TODO() 함수가 있습니다. 반환 타입은 Nothing이므로 예상 타입에 관계없이 사용할 수 있습니다. 이유 파라미터를 허용하는 오버로드도 있습니다.

fun calcTaxes(): BigDecimal = TODO("Waiting for feedback from accounting")

IntelliJ IDEA의 Kotlin 플러그인은 TODO()의 의미를 이해하고 TODO 도구 창에 코드 포인터를 자동으로 추가합니다.

다음 단계는 무엇입니까?