본문 바로가기
반응형

Android 개발231

ifEmpty 와 orEmpty 에 대해서 알아보자 # Kotlin 오늘은 Kotlin 의 ifEmpty 와 orEmpty 에 대해서 정리해 보도록 하겠습니다. 1. ifEmpty Kotlin에는 emptyList인 경우에 디폴트 값을 넘겨줄 수 있는 API가 있는데요. 바로 ifEmpty()입니다. 이 함수의 코드를 보면 아래와 같은데요. isEmpty()로 Collection이 비어있는지를 확인하고, 비어있다면 디폴트 값을 넘겨줍니다. 2023. 5. 30.
CoerceIn, coerceAtMost, coerceAtLeast 범위지정 함수에 대한 정리 # Kotlin 오늘은 Kotlin 에서 값의 범위를 강제할 수 있는, coerceIn, coerceAtMost 그리고 coerceAtLeast 에 대해서 정리해 보겠습니다. 1. CoerceIn 원래 Coerce 라는 단어는 강제로 무언가를 하게 한다는 의미인데요. CoerceIn 은 강제로 어느 범위로 넣게 하겠다는 것 입니다. 인자로 최소값과 최대값을 받는데요. 이 최소값보다 적게나오면, 최소값을, 최대값보다 크게나오면, 최대값을 리턴하도록 강제해주는 API 입니다. 예를 들어서 아래 코드와 같은 경우를 들 수 있습니다. 아래는 Percent 값을 구하는데, distance나 distanceBetween 이 마이너스가 나올 경우에도, 0과 1 사이에서만 값을 주도록 강제해 주었습니다. val distanceInPe.. 2023. 5. 26.
EOFException 발생시키는 empty body 대응 방법 # Retrofit Retrofit 을 이용해서 정상적으로 서버의 데이터를 요청하였을 때, 200 OK를 받았는데도, 서버에서 empty body를 주어서, EOFException 이 발생하는 경우가 있습니다. 오늘은 이것의 대응 방법에 대해서 정리해 보겠습니다. 1. 서버 200 과 EOFException 위에서 언급한 것처럼, 서버에서 200 OK 를 주었는데도, body가 비워져 있어서, EOFException 이 발생할 수 있습니다. 서버에서는 데이터 없으니 "" 를 보내버린 것이구요. 아래와 같이 body는 0-byte가 되어 있습니다. 2023. 5. 25.
MapNotNull 과 MapNotNullTo 에 대한 정리 # null 제거 오늘은 MapNotNull 과 MapNotNullTo 에 대해서 정리해 보도록 하겠습니다. 1. MapNotNull mapNotNull은 엘리먼트에 주어진 transform 연산자를 적용하여서, null 이 나오지 않는 값들로만 list 를 구성해서, 반환해주는 함수입니다. 이름에 map이 있지만, 반환하는 타입은 아래와 같이 List 입니다. null 을 제거해주는 방어연산자라고도 할수 있겠습니다. 2. MapNotNull 예제1 코드를 보면서 이해해 보겠습니다. mapNotNull연산자 안에서, 2로 나누어 나머지가 없이 딱 떨어지는 값은 2를 곱해주고, 그렇지 않은 경우는 null을 리턴하는 조건문을 넣어주었습니다. val numbers = listOf(1, 2, 3, 4, 5, 6) val dou.. 2023. 5. 24.
Broadcast Receiver 등록하고 시스템 이벤트 받아서 처리하기 안드로이드의 시스템은 이벤트가 발생할 때 broadcast를 보내줍니다. 이름이 broadcast이니, 방송을 한다고 해야할까요. 안드로이드 내부적으로 이벤트가 발생을 했다고 알려주는 것인데요, 안드로이드 앱에서, 시스템에게 특정한 이벤트에 관해 관심이 있다고 등록을 하면, broadcast를 수신할 수 있게 되는데요. 오늘은 사용자가 이어폰을 스마트폰에서 빼거나 꽂았을 때, 시스템 이벤트를 받아서 토스트 팝업을 띄우는 방법에 대해서 정리해보겠습니다. 네이버 뮤직 같은 앱들이 헤드폰을 빼거나 꽂았을 때, 플레이를 멈추도록 하는 기능도 broadcastReceiver를 이용했을 거라고 생각이 됩니다. 1. BroadCast 받는 방법 안드로이드의 시스템으로부터는 여러가지 broadcast를 받아서 볼 수 .. 2023. 5. 23.
Vibrator 로 구현하는 안드로이드 진동 앱을 통해 사용자에게 알리는 방법중 진동은 매우 중요한 수단중 하나인데요. 오늘은 Vibrator API를 이용해서 이를 구현하는 방법에 대해서 정리해보겠습니다. 1. Manifest 설정 제일 먼저 해주어야 하는것은 permission을 얻는 것인데요. 다행히도 동적으로 얻어야 하는 수준의 permission은 아닙니다. 아래와 같이, manifest 에서 uses-permission만 VIBRATE에 대해서 정의해주면 됩니다. 2. Vibrator 객체 얻기 가장 먼저 할 일은 Vibrator객체를 얻어오는 것 입니다. 안드로이드 하드웨어와 관련된 클래스들이 그렇듯이, getSystemService 메소드를 사용해서 객체를 얻어와야 합니다. API23 버전(버전 M)부터는 아래와 같이 얻어오구요. 그.. 2023. 5. 23.
List 아이템 부분추출 함수 정리 # take, takeWhile, takeLast, drop, slice, first and last 오늘은 list 나 Sequence에 적용할 수 있는, 아이템을 부분적으로 추출하는 방법에 대해서 정리해 보도록 하겠습니다. 오늘 볼 API에는 다음과 같은 것들이 있는데요. take, takeLast, takeWhile, drop, dropWhile, slice first, last, firstOrNull, lastOrNull 하나씩 보도록 하겠습니다. 1. take api 이름에서 느껴지는 것처럼, 처음 n개의 요소들을 반환해 줍니다. 아래 코드에서는 처음3개를 취하기 위해서, 3을 인자로 넣어주었구요. 결과는 앞의 1,2,3을 반환해 주게 됩니다. val numbers = listOf(1, 2, 3, 4, 5) val result = numbers.take(3) println(result) // 결.. 2023. 5. 15.
compareBy 와 min 그리고 sortedWith 사용방법에 대한 정리 오늘은 리스트를 sorting 할 때 사용할 수 있는, compareBy 와 min 그리고 sortedWith를 사용하는 방법에 대해서 정리해 보겠습니다. 1. 최소값을 찾을 때 2가지 조건을 붙이는 방법에 사용 먼저 CompareBy 와 min 을 사용하는 방법에 대해서 정리해 보겠습니다. 아래 코드에서 사용한 것은 minWithOrNull 인데요. 코드를 먼저 보고 정리해 보겠습니다. compareBy에 두가지 조건 블록이 들어가 있습니다. 이 조건은 아래와 같이 적용됩니다. 먼저 value 가 적은 최소값을 찾고, 다음으로 id가 적은 값을 찾는다 입니다. data class SampleData(val id: Int, val value: Int) val dataList = listOf( Sample.. 2023. 5. 10.
any , contains, none , all, containsAll 에 대한 정리 # Kotlin list 존재여부 확인 오늘은 Kotlin list 의 존재여부를 확인하는 함수인, any, contains, none, all, containsAll 에 대해서 정리해 보도록 하겠습니다. 1. any() 적어도 하나 이상의 매칭되는 아이템이 Collection 에 존재한다면, true 를 없으면 false 를 return 해 줍니다. val list = listOf(1, 2, 3, 4, 5) val result = list.any { it > 3 } println(result) // 결과: true 2023. 5. 6.
StateFlow vs SharedFlow 를 비교해보자 #이벤트 핸들링 오늘은 StateFlow 와 SharedFlow 에 대해서 정리해 보도록 하겠습니다. 1. 기존 글 참조 만약 SharedFlow와 StateFlow 각각에 대한 기본적인 부분들에 대해서 정리하고 싶으시다면, 아래 글들을 참조하신 다음에 이 글을 읽으면 더욱 도움이 될 것 입니다. >> SharedFlow 에 대한 총정리 # Buffer Replay tryEmit Kotlin Coroutine >> mutableStateOf 와 MutableStateFlow 비교 총정리 # collectAsState 2. StateFlow 는 State 변화를 위한 API 2-1. StateFlow 는 이벤트핸들링을 위한 API가 아니다 아래는 Kotlin 이 아닌 Android의 공식문서인데요. 중요한 부분이 잘 설명.. 2023. 5. 6.
SharedFlow 에 대한 총정리 # Buffer Replay tryEmit Kotlin Coroutine 오늘은 Kotlin Coroutine의 SharedFlow 에 대해서 정리해 보도록 하겠습니다. 1. SharedFlow SharedFlow 는 이름에서 알 수 있듯이, Collector 가 여러개인 경우, Collector 들이 emit 된 값들을 동시에 consume 할 수 있도록, Share(공유)되는 Flow 의 API 입니다. fun main() = runBlocking { val sharedFlow = MutableSharedFlow(replay = 1) launch { for (i in 1..5) { sharedFlow.emit(i) println("Emitted: $i") } } launch { sharedFlow.collect { value -> println("Collector 1에서 .. 2023. 5. 4.
Flow 결합연산자 combine , zip , merge 비교 총정리 # Kotlin Coroutine 오늘은 Kotlin Coroutine의 Flow API 중, flow들을 결합해 주는 오퍼레이터인, combine과 zip 그리고 merge에 대해서 총정리해 보겠습니다. 1. combine Combine 은 2개 이상의 스트림 되는 flow 데이터들을, 합쳐서 하나의 flow로 흘려보내 주는 API입니다. 2개의 스트림을 합쳐주는 것은 알겠는데, 각각의 데이터들이 다른 시점에 나올 때는 어떻게 되는 것일까요? 아래 공식문서의 설명에 따르면, 둘 중 가장 최근에 방출되는 값이 있으면, 그 값을 기준으로 합쳐진 flow 를 방출해 줍니다. 단순히 2개의 flow를 합쳐준다고만 이해해서는 안 되고요. 2개의 flow 중 어느 하나에서 아이템이 방출되면, 최근 2개의 flow를 합쳐서 방출해 줍니다. 좀 더.. 2023. 5. 3.
onEach vs onStart 비교 정리 # Kotlin Coroutine Flow 오늘은 Kotlin Coroutine Flow의 onEach 와 onStart 에대해서 정리해 보도록 하겠습니다. 1. onEach flow 에서 아이템이 흘러나올 때 마다, 실행하도록 해야하는 코드가 있다면, 이 연산자를 사용하는 것이 좋습니다. upstream 즉 위의 flow 에서 흘러나오는 값을 그대로 전달해 주기 때문에, 흘러나가는 데이터에는 영향을 주지 않습니다. 코드를 보면서 이해해 보도록 하겠습니다. val flow = flowOf(1, 2, 3, 4, 5) flow .onEach { item -> println("Item emitted: $item") } .map { item -> item * 2 } .collect { result -> println("결과: $result") } 2023. 5. 3.
Destructuring declaration 에 대해 알아보자 # 구조분해 선언 Kotlin 오늘은 Destructuring declaration 에 대해서 정리해 보도록 하겠습니다. 1. Destructuring declaration 한국말로는 구조파괴 또는 분해 선언이라고 할 수 있는데요. 단 한줄로, 객체의 여러 프로퍼티들을 한번에 선언하게 해주는 문법입니다. 코드가 짧아지고, 가독성도 향상됩니다. 2. 구현 2-1. 구현준비 실제로 코드로 보면서 이해하는 것이 빠를텐데요. 다음과 같은 간단한 Student 클래스가 있다고 가정해 보겠습니다. data class Student(val name: String, val age: Int) 이제 위의 데이터 클래스로 객체를 아래와 같이 만들어줍니다. val student = Student("Ive Shim", 23) 2023. 5. 1.
Kotlin Escape 에 대한 정리 # RawString Escaped String Literal 오늘은 Kotlin 의 String Literal 에서 사용되는 Escape 규칙에 대해서 정리해 보겠습니다. 1. Escape Kotlin String Literal 에서 Escape는 "\" 를 이용해주면 되는데요. 원래는 그냥 사용하면 제대로 전달이 안되지만, 아래와 같이 사용해주면 그대로 전달될 수 있습니다. 특히나 따옴표(Single Quote)나 쌍따옴표(Double Qoute) 같은 경우 꼭 알아두어야 합니다. 기호 의미 \" Double quote \' Single quote \\ Backslash \n Newline \r Carriage return \t Tab 2023. 4. 30.
Kotlin GroupBy 구현과 정리 # List 그룹핑 오늘은 Kotlin의 GroupBy 에 대해서 정리해 보겠습니다. 1. GroupBy API 우선 해당 API의 코드를 가볍게 보고 가도록 하겠습니다. 아래에서 보듯이 GroupBy는 Iterable 타입의 collection 에서, 전달되는 특정한 기준을 Map의 Key값으로 해서, 그에 해당하는 데이터들을 List로 묶어서 Value에 넣어줍니다. 2023. 4. 28.
getOrNull 과 getOrElse 에 대한 정리 # List Kotlin Kotlin에는 편리한 함수들이 정말 많은데요. 오늘은 Kotlin 의 getOrNull 과 getOrElse 에 대해서 정리해 보도록 하겠습니다. 1. getOrNull () 과 getOrElse() 1-1. getOrNull 주어진 Index에 대해서 엘리먼트가 있으면 반환해주고, 해당 index가 범위내에 없으면 null 을 반환해 주는 Kotlin Collections 의 list 입니다. 2023. 4. 28.
SharedPreference 로 간단한 데이터 저장하기 # Android 안드로이드 개발을 하면서 간단한 key-value 형태의 값들을 앱안에 저장할 때가 있는데요. 예를 들면, 앱의 글자 크기설정, 알람 온오프 같은 값입니다. 그런데, 이럴때마다 DB를 사용하는 것은 너무나 무거운 일인데요. SharedPreference를 사용하면 좀더 가볍고, 쉽게 해결할 수 있습니다. 1. SharedPreference 는? SharedPreference는 비교작 작은 크기의 키-밸류값을 읽고 쓸수 있도록 Android Framework에서 제공해주는 기능입니다. 위에서 애기한것처럼, 설정 값등을 저장 하는데 많이 사용 하구요. 경우에 따라서 private하게 사용하거나, share될 수 있도록 할 수 있습니다. 2. SharedPreference 초기화 먼저 SharedPrefere.. 2023. 4. 27.
Kotlin Pair 와 Map 함수 이용해서 데이터 가공하기 오늘은 Kotlin 에서 Pair()와 Map() 함수를 이용하는 방법에 대해서 정리해 보겠습니다. 1. Pair 데이터 먼저 아래와 같은 Pair()객체를 list 에 여러개 있다고 가정해 보겠습니다. val testList = listOf(Pair("one", 1), Pair("two", 2), Pair("three", 3)) 참고로 to 키워드를 사용하면 Pair()생성자 형태를 사용하지 않고도 Pair 객체를 만들수 있습니다. Kotlin에 있는 infix함수라는 개념인데, 보통 사용하는 점(.)이나 괄호() 를 사용하지 않고 함수를 호출하게 해줍니다. val testList = listOf("one" to 1, "two" to 2, "three" to 3) 2023. 4. 26.
Room DB 에서 검색 구현하기 # Like ExactMatch SQL 오늘은 RoomDB에서 검색하기를 구현하는 방법에 대해서 정리해 보겠습니다. entity, dao 등 RoomDB에 관한 기본적인 내용은 아래 글을 참조해 주세요. >> Room DB 사용방법 총정리 # Android SQLite 1. 검색을 위해 준비한 데이터 검색을 구현하기 전에 먼저 데이터가 저장되어 있어야 하는데요. 아래와 같이 아주 간단한 데이터가 있다고 가정해 보겠습니다. 표에는 안적혀 있으나, Database명은 "student" 로 하였습니다. studentId Name 1 Aileen lee 2 Eyenie park 3 Ive Kim 2. 검색을 위한 DAO 작성 2-1. 단순한 like 검색 위의 데이터에서 이름으로 학생데이터를 검색하기 위해서는 아래와 같이 Dao를 작성해 주어야 합니.. 2023. 4. 24.
getLaunchIntentForPackage 로 다른 앱을 실행 하는 방법 # query Android 11 오늘은 안드로이드 앱에서 다른 앱을 launch 시키는 방법을 정리해 보도록 하겠습니다. 1. Query Manifest 적용 Android11(API30)이 발표된 이래로, 앱내에서 다른 앱을 실행하는 경우 Manifest에서, 다른 앱에 대한 패키지 명을 명시해 주어야만 합니다. 예전에는 없었던 작업이므로, 간혹 오래된 앱들은 이것이 선언되지 않아서, 기능이 작동하지 않을 수 있습니다. 혹시 잘 되던 기능이 않되고 있다면 이 부분을 체크해 주어야 합니다. manifest 에 선언하는 것이므로, Enum 클래스를 사용할 수 없습니다. 하드코딩해야 하므로, 테스트를 반드시 해 주어야 합니다. ... ... 2023. 4. 20.
Coroutine suspend 동작에 관한 좋은 예와 잘못된 예 # 비동기 오늘은 Coroutine 은 기본적으로 비동기를 기본으로 합니다. 그렇기에 비동기 블록이 suspend 되어서 결과값을 받아서 실행해야 할 경우는, suspend 함수를 위치시킬 때 생각을 해야 합니다. 그렇지 않으면 예상치 못한 동작을 보게되는데요. 오늘은 suspend 함수를 이용해서 Coroutine을 잘 사용한 예와 잘못 사용한 예를 점검해 보겠습니다. 1. suspend 가 제대로 동작하지 않는 잘못된 케이스 우선은 코드를 보도록 하겠습니다. 아래에서 launch 블록바깥에 결과를 얻어서, print 하는 것이 의도인데요. 실제로 실행해 보면, result에 값이 담기기 전에 지나쳐 버립니다. suspend fun fetchData(): String { delay(2000) return "dat.. 2023. 4. 18.
flatMapLatest 이용해서 값이 들어오는 것을 기다리기 # Coroutine 오늘은 flapMapLatest 를 활용해서 특정한 값이 들어오는 것을 기다리다가, 값이 들어오면 특정 코드를 실행시키는 방법에 대해서 정리해 보겠습니다. 물론, Coroutine 의 supsend 함수를 이용하면 어려운 일은 아니지만, 만약, Global 한 변수에 들어오는 값에 대해서라면 조금 다른 접근이 필요하기 때문입니다. 1. FlatMapLatest 이 API는 Kotlin Coroutine 의 Flow API인데요. Flow에서 흘러나오는 데이터 스트림에서, 가장 최근의 값만 취하기 위해서 만들어 졌습니다. 새로운 값이 흘러나오면 기존 flow는 cancel 시켜버리도록 되어 있습니다. 2. 값이 들어오는 것 기다리기 구현 가장 최근 값이 들어오는 것을 기다릴 때도, flatmapLates.. 2023. 4. 18.
GeneratedInjector could not be resolved 에러 대응 방법 # package name 아주 오래된 프로젝트를 다시 빌드해 보니, hilt 와 관련해서 아래와 같은 에러로그를 보게 되었습니다. Fragment 에서 발생한 에러였는데, Hilt 에 관련한 셋업은 문제가 없었고, @AndroidEntryPoint 를 붙이는 것과 같은 기본적인 사항도 문제가 없었습니다. public abstract static class FragmentC implements TestFragment_GeneratedInjector, ^ symbol: class TestFragment_GeneratedInjector location: class MyApplication_HiltComponents error: ComponentProcessingStep was unable to process 'com.exam.test.. 2023. 4. 15.
함수안에 함수 넣기 # Closure Local functions Kotlin 오늘은 Kotlin에서 함수안에 함수를 넣는 것에 대해서 정리해 보도록 하겠습니다. 1. 함수안에 함수 코틀린에서는 local function 즉, 함수안에 다른 함수를 넣는 것이 가능합니다. fun main() { fun hello(name: String) { println("Hello, $name!") } hello("김군") hello("홍군") } 2. Clousure 함수안에 함수를 사용할 때는 Closure라는 개념이 들어갑니다. 쉽게 말해서, 함수 바깥의 범위에 들어가 있는 변수의 값에 접근하는 것이 가능하다는 것 입니다. 아래의 calculateDiscount 함수는 바깥에 있는 price와 discountPercentage에 접근이 가능한 것도 이것 때문입니다. 원래는 global한 범위.. 2023. 4. 15.
데이터보안 양식 잘못됨 SPLIT_BUNDLE 13 정책 선언 # Google Play 가끔 사용자정보를 수집하지 않고 있는데, Google 에서 정책준수 위반 메일을 받으신 분들이 있습니다. 오늘은 이렇게 데이터보안 양식 관련 정책 위반을 받고 앱게시 중단에 대해서 정리해 보겠습니다. 1. SPLIT_BUNDLE 13: 정책 선언 어느 날 이런 메일을 받게 됩니다. 내용을 정리하면, 사용자 정보를 수집하는데, 안 하고 있다고 하였다. 그러니 수정하라는 것 입니다. 2023. 4. 14.
함수 실행 시간 측정 후 Delay 사용하기 오늘은 함수의 실행시간을 체크하여서, 특정시간을 지나지 않으면, 그 만큼 Delay를 시키는 코드를 정리해 보겠습니다. 이런 코드는 특히 Splash화면 진행중에 하기에 용이한데요. 예를 들어 Splash를 3초정도 보여주고 싶다면, 특정 코드를 실행시키고 3초에서 남은 시간을 측정한 후, 나머지 시간만큼 Delay를 부여해 준다음 화면을 보여주기만 하면 되겠지요. 1. 함수 실행시간 측정하기 함수의 실행시간 측정은 아래와 같이 할 수 있습니다. 먼저 deleteFiles()라는 함수를 실행하였고요. 바로 직전과 실행후의 시간을 측정합니다. 시간 측정은 system.currentTimeMillis 를 활용하였습니다. 아래와 같은 코드를 이용하면, 최대 시간인 maxTimeMillis로부터, 걸린 시간(e.. 2023. 4. 14.
안드로이드 앱 내부 파일 확장자 체크 후 삭제하기 # delete filesDir 안드로이드에서는 앱에서 사용할 디렉토리를 따로 정해주었는데요. 이러한 내부에 파일들에 사용하지 않는 이미지들을 저장했다가 지워야할 때가 있습니다. 앱 내부의 디비에서는 이미지 uri를 가지고 있다가 유저가 해당 데이터를 지웠는데, 이 때, 내부 디렉토리의 파일로는 남아있는 경우입니다. 1. FilesDir 와 CacheDir 안드로이드에서는 내부파일에 접근하기 위한 장소로 FilesDir를 제공해주고 있는데요. 이곳은 완벽하게 다른 앱들은 접근할 수 없는 앱만의 영역으로, 앱이 필요로 하는 데이터나 캐쉬파일들을 저장할 수 있습니다. API29버전 이상부터 이 영역은 암호화 되어서 저장되게 됩니다. 그리고 CacheDir은 이름에서도 알 수 있듯이, 앱에서 사용할 캐쉬파일들을 저장하는 곳 입니다. 당연히 .. 2023. 4. 14.
Kotlin custom getter 와 setter 구현하기 Kotlin 에서는 Custom 하게 getter 와 setter 를 구현할 수 있는데요. 오늘은 이것에 대해서 알아보고, 활용 사례까지 정리해 보겠습니다. 1. Custom Getter 와 Setter Kotlin은 우리가 특별히 무엇을 하지 않아도, 알아서 getter 와 setter 를 만들어 줍니다. Java에서는 class의 길이가 길었었는데, 덕분에 Kotlin에서는 매우 간결해 졌습니다. 그런데, 우리는 때로 override 해서 다른 방식으로 get하거나 set을 해야할 때가 있는데요. 이것을 하는 것을 보도록 하겠습니다. 1-1. Custom Getter 아래와 같이 age를 정의할 경우, custom하게 getter를 설정하려면 아래와 같이 해주면 됩니다. get()을 넣어주고, 수식으.. 2023. 4. 11.
isNullOrEmpty 와 isNullOrBlank 의 차이점 # Kotlin 오늘은 Kotlin의 isNullOrEmpty 와 isNullOrBlank 의 차이점에 대해서 정리해 보겠습니다. 둘은 비슷하기 때문에 잘 알지 않고 사용하는 경우도 있는데요. 오늘 차이를 구분해 보도록 하겠습니다. 1. isNullOrEmpty 이 API는 만약 String이 null 이거나 empty 일 경우 true를 반환해 준다고 설명되어 있습니다. 그런데, empty는 어떤 것을 말하는 것일까요? 비어있다는 뜻 입니다. 즉 ""을 가리킵니다. String 값에 대해서 ""로 초기화하는 경우도 있는데요. 그대로 아무 값이 없는지 보려면 isNullOrEmpty로 충분합니다. val xInitValue: String? = "" println(xInitValue.isNullOrEmpty()) // t.. 2023. 4. 9.