billcorea.com
빌코리아의 홈페이지 입니다.
2026/02/26
오늘의 이야기
#스하리1000명프로젝트
스치니들!
내가 만든 이 앱은, 내 폰에 오는 알림 중에서 중요한 키워드가 있는 경우
등록해둔 친구에게 자동으로 전달해주는 앱이야 📲
예를 들어, 카드 결제 알림을 와이프나 자녀에게 보내주거나
이번 달 지출을 달력처럼 확인할 수도 있어!
앱을 함께 쓰려면 친구도 설치 & 로그인해줘야 해.
그래야 친구 목록에서 서로 선택할 수 있으니까~
서로 써보고 불편한 점 있으면 알려줘 🙏
👉 https://play.google.com/store/apps/details?id=com.nari.notify2kakao
오늘의 이야기
이전 부터 해오던 취미 생활을 업그레이드 해야 할 것 같아요... 그래서 다시금 자료를 찾아 수집을 해 볼 까 합니다.
오늘은 첫번째... 영어로 작성된 글을 옮겨 적어 봅니다. 원작자 분에게 감사의 인사를 드리며...
https://cruzc09.medium.com/setting-up-a-rasberry-pi-4-19617ec1f709
Setting up a Rasberry Pi 4
Although there are several Rasberry Pi models, it is without a doubt that the Raspberry Pi 4 Model B is the newest, cheapest, and simplest…
cruzc09.medium.com
Rasberry Pi 모델이 여러 개 있지만 의심할 여지없이 Raspberry Pi 4 Model B가 가장 최신이고 가장 저렴하고 단순한 것입니다. 2GB에서 8GB까지 가능합니다. 작동하는 프로토타입을 얻은 후에는 더 작고 제어가 덜 필요한 Raspberry Pi Zero로 이동할 수 있습니다.
이 가이드는 초기 설정 과정을 안내합니다.
USB C & Power Supply
Raspberry Pi 4에는 USB C를 통해 최소 3.0A를 제공하는 전원 공급 장치가 필요합니다. Raspberry는 또한 전원 공급 장치 충전기도 제공합니다. (이전 Raspberry Pi 3 에서는 Micro 5pin type 의 전원 공급 장치를 이용했습니다.)
microSD Card
Raspberry Pi OS 운영 체제를 저장하려면 Raspberry Pi에 최소 8GB 공간의 SD 카드가 필요합니다. Samsung Evo 256GB는 속도와 볼륨으로 인해 개인적으로 가장 좋아하는 제품입니다. (이전 버전에서는 64GB 정도 였을 것 같은데...)
Keyboard and Mouse
유선으로 연결된 Raspberry Pi를 사용하려면 USB 키보드와 USB 마우스가 필요합니다. Logitech Media Combo를 선택했습니다. ( 그냥 저렴한 키보드 세트를 활용 해도 됩니다.)
Monitor
Raspberry Pi 4의 멋진 점 중 하나는 두 개의 개별 모니터를 연결할 수 있는 두 개의 마이크로 HDMI 포트가 있다는 것입니다. 따라서 마이크로 HDMI-HDMI 케이블이 필요합니다. (이전 버전에서는 한 개만 지원이 되었는데요)
Rasberry Pi Imager
시스템(Windows 또는 Mac)에 적합한 Rasberry Pi Imager를 여기에서 다운로드하십시오. Rasberry Pi Foundation은 훌륭한 일을 해냈습니다. 이미 저는 Python, Java, Sonic 등으로 SD 카드를 설정합니다. (이전 버전에서 Python 을 설치해서 자료 수집을 매일 하고 있기도 합니다.)
Start your Rasberry Pi
이제 SD 카드를 Raspberry Pi에 마운트하고 전원을 켤 수 있습니다. 기본 사용자 이름은 pi이고 암호는 raspberry입니다.
(사용자 비밀번호는 변함이 없네요...)
이렇게 읽어보니 그닥 어려운 이야기는 아닌 듯 합니다.
이 별그램은 제가 Raspberry Pi 3 에서 python 스크립트를 이용해서 이미지만 포스팅 하는 계정이기도 합니다.
오늘도 즐~ 좋은 밤 되세요...
오늘의 이야기
https://medium.com/androiddevelopers/workmanager-kotlin-apis-a0fb9dfbfeb6
WorkManager — Kotlin APIs
WorkManager provides a set of APIs that makes it easy to schedule asynchronous tasks for immediate or deferred execution that are expected…
medium.com
이제 kotlin 으로 전향(?)을 하기 위해서 배우기 시작...
오늘은 medium 에서 보내온 메일 내용 중에 workmanger 관련된 자료가 있어서 번역을 해 볼까 한다.
먼저 글을 작성해 주신 분께... 감사합니다.
먼저 workManager 에 대해서 이해을 해야 한다면... 작업관리자 라고 이해를 하면 되지 않을 까 한다. 나의 앱에서 반복적인 작업을 해야 하는 경우, 작업관리자를 통해서 일회성, 주기적인 반복작업을 나의 앱에 일을 시킬 수 있게 된다.
이제 어떻게 저자는 설명을 하는 지 읽어보고, 이해를 해 보도록 하겠다.
--- 번역글 ---
WorkManager — Kotlin API
WorkManager 는 앱이 닫히거나 장치가 다시 시작되는 경우에도 실행될 것으로 예상되는 즉시 또는 지연된 실행을 위한 비동기 작업을 쉽게 예약할 수 있는 API 세트를 제공합니다. Kotlin 사용자의 경우 WorkManager는 kotlin 에 대한 최고 수준의 지원을 제공합니다. 이 게시물에서는 WorkManager codelab 을 기반으로 하여 kotlin 이 있는 WorkManager의 기본 사항을 보여 드리겠습니다.시작하겠습니다!
WorkManager의 기본
WorkManager 라이브러리는 사용자가 특정 화면에서 벗어나거나 사용자가 애플리케이션을 백그라운드로 설정하거나 장치를 다시 시작하더라도 계속 실행되어야 하는 모든 작업에 권장되는 선택입니다. 일반적인 작업은 다음과 같습니다.
- 로그 업로드 또는 보고 데이터
- 이미지에 필터 적용 및 이미지 저장
- 주기적으로 로컬 데이터를 네트워크와 동기화
사용자가 화면과 같은 특정 범위를 벗어나면 즉각적인 작업이 종료될 수 있는 경우 Kotlin Coroutine을 직접 사용하는 것이 좋습니다.
WorkManager 코드 랩은 이미지를 흐리게 처리하고 결과를 디스크에 저장합니다. 이를 달성하기 위해 무엇이 필요했는지 봅시다.
work-runtime-ktx종속성을 추가했습니다.
implementation "androidx.work:work-runtime-ktx:$work_version"
우리는 우리 자신의 Worker 클래스를 구현하는 것으로 시작했습니다. 여기에 백그라운드에서 수행하려는 실제 작업에 대한 코드를 넣습니다. Worker 클래스를 확장하고 doWork() 메서드를 재정의합니다. 이것은 가장 중요한 수업이므로 나중에 자세히 다루겠습니다. 초기 구현은 다음과 같습니다.
/* Copyright 2020 Google LLC.
SPDX-License-Identifier: Apache-2.0 */
class BlurWorker(ctx: Context, params: WorkerParameters) : Worker(ctx, params) {
suspend fun doWork(): Result {
val resourceUri = inputData.getString(KEY_IMAGE_URI)
return try {
if (resourceUri.isNullOrEmpty()) {
Timber.e("Invalid input uri")
throw IllegalArgumentException("Invalid input uri")
}
val outputData = blurAndWriteImageToFile(resourceUri)
Result.success(outputData)
} catch (throwable: Throwable) {
Timber.e(throwable, "Error applying blur")
Result.failure()
}
}
…
}그런 다음 작업 요청을 작성합니다. 우리의 경우 작업을 한 번만 수행하기를 원하므로 OneTimeWorkRequest.Builder. 입력 Uri으로 흐리게 처리하려는 이미지를 설정합니다.
Kotlin 팁: 입력 데이터 workDataOf를 생성하기 위해 데이터 빌더를 생성하고 키-값 쌍을 넣고 데이터를 생성하는 함수를 사용할 수 있습니다.
/* Copyright 2020 Google LLC.
SPDX-License-Identifier: Apache-2.0 */
val blurBuilder = OneTimeWorkRequestBuilder<BlurWorker>()
val data = workDataOf(KEY_IMAGE_URI to imageUri.toString())
blurBuilder.setInputData(data)작업을 예약하고 실행하기 위해 WorkManager 클래스를 사용합니다. 우리는 수행해야 할 작업과 이러한 작업에 대한 제약 조건을 제공할 수 있습니다.
/* Copyright 2020 Google LLC.
SPDX-License-Identifier: Apache-2.0 */
val workManager = WorkManager.getInstance(application)
val continuation = workManager.beginUniqueWork(blurBuilder.build())
// Actually start the work
continuation.enqueue()작업자가 작업을 수행하도록 하십시오.
를 사용하면 WorkerWorkManager가 자동으로 Worker.doWork()백그라운드 스레드를 호출합니다. Result반환된 에서 작업 doWork()이 성공했는지 여부와 실패한 경우 작업을 다시 시도해야 하는지 여부를 WorkManager 서비스에 알립니다.
Worker.doWork() 동기 호출입니다. 차단 방식으로 전체 백그라운드 작업을 수행하고 메서드가 종료될 때까지 완료해야 합니다. doWork()에서 비동기 API를 호출하고 결과를 반환하면 콜백이 제대로 작동하지 않을 수 있습니다.
하지만 비동기 작업을 하려면 어떻게 해야 할까요?
예제를 복잡하게 만들고 데이터베이스에서 흐리게 처리된 모든 파일의 Uris를 저장하려고 한다고 가정해 보겠습니다.
이를 위해 다음을 만들었습니다.
- 간단한 BluredImage 엔터티
- 이미지를 삽입하고 가져오는 dao 클래스
- 데이터베이스
여기에서 구현을 확인하십시오.
데이터베이스에 데이터를 저장하거나 네트워크 요청을 수행하는 것과 같은 비동기 작업을 수행해야 하는 경우 Kotlin에서 CoroutineWorker.
CoroutineWorker를 사용하면 Kotlin 코 루틴을 사용하여 비동기 작업을 수행할 수 있습니다.
doWork() 방법은방법 suspend입니다. 그래서 이것은 우리가 여기에서 정지된 도를 쉽게 호출할 수 있음을 의미합니다.
/* Copyright 2020 Google LLC.
SPDX-License-Identifier: Apache-2.0 */
class BlurWorker(ctx: Context, params: WorkerParameters) : CoroutineWorker(ctx, params) {
override suspend fun doWork(): Result {
val resourceUri = inputData.getString(KEY_IMAGE_URI)
return try {
if (resourceUri.isNullOrEmpty()) {
Timber.e("Invalid input uri")
throw IllegalArgumentException("Invalid input uri")
}
val outputData = blurAndWriteImageToFile(resourceUri)
// save uri in the database
val imageDao = ImagesDatabase.getDatabase(applicationContext).blurredImageDao()
imageDao.insert(BlurredImage(resourceUri))
Result.success(outputData)
} catch (throwable: Throwable) {
Timber.e(throwable, "Error applying blur")
Result.failure()
}
}
...
}기본적으로 doWork()를 사용합니다 Dispatchers.Default. 필요한 Dispatcher로 이를 재정의할 수 있습니다. 우리의 경우 Room이 이미 삽입 작업을 다른 Dispatcher로 이동하기 때문에 이 작업을 수행할 필요가 없습니다. 자세한 내용은 Room Kotlin API 게시물을 확인하세요.
CoroutineWorker사용자가 앱을 닫아도 완료해야 하는 비동기 작업을 수행하려면 사용을 시작하세요.
WorkManager에 대해 더 자세히 알고 싶다면 이를 심층적으로 다룰 향후 시리즈를 계속 지켜봐 주십시오. 그때까지 코드 랩과 문서를 확인하세요.
- WorkManager 문서
- WorkManager 코드랩 작업
- 고급 WorkManager 코드랩
--- 번역글 끝 ---
작성자님의 노고(?)에 감사드리면서... 덕분에 잘 배워 보도록 하겠습니다.
오늘의 이야기
#스하리1000명프로젝트,
कभी-कभी विदेशी कामगारों से बात करना कठिन होता है, है न?
मैंने एक सरल ऐप बनाया है जो मदद करता है! आप अपनी भाषा में लिखते हैं, और दूसरे इसे अपनी भाषा में देखते हैं।
यह सेटिंग्स के आधार पर स्वचालित अनुवाद करता है।
आसान चैट के लिए बहुत उपयोगी। जब मौका मिले तो देख लेना!
https://play.google.com/store/apps/details?id=com.billcoreatech.multichat416
오늘의 이야기
https://developer.android.com/google/play/billing/billing_reference?hl=ko
Google Play 결제 AIDL 참조 문서 | Google Play 결제 시스템 | Android Developers
이 문서에서는 Google Play 결제 AIDL을 사용하기 위한 기술 참조 정보를 제공합니다.
developer.android.com
BILLING_RESPONSE_RESULT_ITEM_ALREADY_OWNED (응답 코드 7)를 만나는 경우는 어떻게 할까?
fun getAllPurchasedItem() {
billingClient.queryPurchaseHistoryAsync(BillingClient.SkuType.INAPP, this)
}
// 최근 구매한 아이템을 알고자 할 때 사용
// getAllPurchasedItem 의 this
override fun onPurchaseHistoryResponse(
billingResult: BillingResult,
purchaseHistoryList: MutableList<PurchaseHistoryRecord>?
) {
Log.e("onPurchaseHistoryResponse", "${billingResult.debugMessage}" )
if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
if (!purchaseHistoryList.isNullOrEmpty()) {
// 가장 최근에 구매된 아이템을 확인할 수 있다.
purchaseHistoryList.forEach {
Log.e("onPurchaseHistoryResponse", "Previous Purchase Item : ${it.quantity}")
// 소비 되지 않은 것이 있다면 소비 처리 하기
var consumeParams = ConsumeParams.newBuilder()
.setPurchaseToken(it.purchaseToken)
.build()
billingClient.consumeAsync(
consumeParams,
) { billingResult,_ ->
if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
Log.e("onPurchaseHistoryResponse", "${billingResult.debugMessage}")
log(msg="${billingResult.debugMessage}")
}
}
}
}
}
}쉬운 방법은 이전 구매 history을 열어 보는 방법이 있고, 그리고 이미 구해했던 이력을 만나게 되면 구매 이력을 이용해서 소비를 해 버리는 방법으로 해소를 하였다.
구글링을 해서 찾아보다 보면 이런 상태가 되는 이유는 구매 처리를 진행하는 동안 네트워크가 끊어져서 처리가 정상적으로 되지 않았을 경우라고 하고 있기는 하지만, 나의 경우는 앱을 개발하는 동안 구매 처리를 했는 데, 소비 처리를 하지 않았더니, 계속 그런 현상이 발생하였다. 그래서 찾은 방법이 구매 이력을 찾아서 모두 다 소비시키는 방식으로 한번 처리를 하고 다시 구매 진행을 하였더니, 기존 구매 이력이 소비되는 것을 확인하였다.
BILLING_RESPONSE_RESULT_ITEM_UNAVAILABLE (오류코드 4) 이런 경우는 앱 개발을 하는 경우 만나게 되는 것은 인앱 결제를 처리하기 위해서는 앱을 출시하기 전에 구매 테스트를 해야 하는 데, 구글 console에 테스트 계정을 등록하여 구매 테스트를 하게 되는 데, AVD의 경우 설정에 들어가서 계정을 한 개만 설정했을 때는 문제가 없지만,
실제 사용 중인 폰의 설정에서 계정을 2개 이상 등록한 경우에는 내 폰에 설정된 계정을 전부다 테스터 계정으로 등록하기 전에는 구매 테스트에서 이런 오류를 만나게 되는 경우가 있을 수 있다.
간간히 발생하는 오류 형태가 실제 개발 중에 테스트 시간을 뺏는 경우가 있을 수 있으므로 기억해 두고 있으면 도움이 될 것 같다.

오늘의 이야기
#billcorea #운동동아리관리앱
🏸 श्नीडल, बैडमिंटन क्लबों के लिए एक आवश्यक ऐप!
👉 मैच खेलें - स्कोर रिकॉर्ड करें और विरोधियों को खोजें 🎉
कहीं भी, अकेले, दोस्तों के साथ, या क्लब में बिल्कुल सही! 🤝
अगर आपको बैडमिंटन पसंद है तो इसे जरूर ट्राई करें
ऐप पर जाएं 👉 https://play.google.com/store/apps/details?id=com.billcorea.matchplay
오늘의 이야기



허재님이 다녀가신 동백꽃수목원
이쁨 다만, 2월엔 꽃이 조금 졌네요.
https://goo.gl/maps/Lp7fY1qQyJpcUy348
Google에서 제공되는 D.Y Kang님의 제주동백수목원 관련 리뷰
★★★★☆ "2월은 꽃이 떨어져서"
www.google.com
오늘의 이야기
#스하리1000명프로젝트,
迷失在韓國?即使您不會說韓語,這個應用程式也可以幫助您輕鬆出行。
只需說出您的語言即可 - 它會翻譯、搜尋並以您的語言顯示結果。
非常適合旅行者!支援英語、日語、中文、越南語等10多種語言。
現在就試試吧!
https://play.google.com/store/apps/details?id=com.billcoreatech.opdgang1127
2026/02/25
오늘의 이야기


섭섭이네 호끌락한집(조그만집)
메뉴는 커리와 국수...
커리는 마침 1인본만 남았다 해서
고기국수 먹음...
그렇게 찾아볼 정도는 아닌 듯
정말 작음 테이블 4개(4인기준)
https://goo.gl/maps/dkjMxHVEHtJSVWhN6
Google에서 제공되는 D.Y Kang님의 섭섭이네 관련 리뷰
★★★☆☆ "그렇게 줄 설 꺼 까지는 없는 것 같아요"
www.google.com
오늘의 이야기
#스하리1000명프로젝트 스치니들! 내가 만든 이 앱은, 내 폰에 오는 알림 중에서 중요한 키워드가 있는 경우 등록해둔 친구에게 자동으로 전달해주는 앱이야 📲 예를 들어, 카드 결제 알림을 와이프나 자녀에게 보내주거나 이번 달 지출을 달력처럼...


