3.9 KiB
3.9 KiB
id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
| id | title | category | status | source_trust_level | verification_status | created_at | updated_at | tags | tech_stack | applied_in | aliases | ||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| native-anr-freeze-debugging | ANR / Freeze — UI 블로킹 추적 | Coding | draft | B | conceptual | 2026-05-09 | 2026-05-09 |
|
|
|
ANR / Freeze Debugging
Android 5초 / iOS 250ms 이상 메인 스레드 블록 = 사용자 체감 freeze. Android = ANR dialog, iOS = Hang. 원인 = 동기 IO / 큰 JSON / Bitmap decode / 락 경합.
📖 핵심 개념
- ANR: Android — Activity 5s, Broadcast 10s, FG service 200s.
- Hang (iOS): MetricKit
MXHangDiagnostic/ Xcode Organizer. - 원인: main thread 에서 disk / network / decode / DB.
💻 코드 패턴
Android — StrictMode (개발용)
class App : Application() {
override fun onCreate() {
super.onCreate()
if (BuildConfig.DEBUG) {
StrictMode.setThreadPolicy(
StrictMode.ThreadPolicy.Builder()
.detectDiskReads()
.detectDiskWrites()
.detectNetwork()
.penaltyLog()
.penaltyDialog() // dev: 다이얼로그
.build()
)
}
}
}
Android — main thread → background
// ❌ main thread DB query
val users = db.userDao().getAll()
// ✅ Coroutine + Dispatchers.IO
viewModelScope.launch {
val users = withContext(Dispatchers.IO) { db.userDao().getAll() }
_state.value = _state.value.copy(users = users)
}
iOS — Main Thread Checker
Xcode → Edit Scheme → Diagnostics → Main Thread Checker (default ON)
실행 중 main 에서 UIKit 외 무거운 작업 시 즉시 stack 출력.
iOS — async 로 빼기
// ❌ main 에서 큰 file decode
let img = UIImage(contentsOfFile: path)
// ✅ background queue
Task.detached(priority: .userInitiated) {
let img = UIImage(contentsOfFile: path)
await MainActor.run { imageView.image = img }
}
Android — ANR trace
adb pull /data/anr/traces.txt
# 또는 Logcat 에서 'ANR in com.app' 검색 → main thread stack 분석
iOS — MetricKit
import MetricKit
class MetricsObserver: NSObject, MXMetricManagerSubscriber {
func didReceive(_ payloads: [MXDiagnosticPayload]) {
for p in payloads {
for hang in (p.hangDiagnostics ?? []) {
print("Hang \(hang.hangDuration): \(hang.callStackTree)")
}
}
}
}
MXMetricManager.shared.add(observer)
Lock 경합
// ❌ main thread synchronized 블록 → IO 가 들어가면 ANR
synchronized(cache) {
cache.put(k, fetchFromNetwork()) // 네트워크 안에서 락 보유
}
// ✅ 락 안에서 short critical section 만
val v = fetchFromNetwork()
synchronized(cache) { cache.put(k, v) }
🤔 의사결정 기준
| 증상 | 도구 |
|---|---|
| 화면 5초 정지 | Android: ANR trace, iOS: Hang Diagnostic |
| 가끔 끊김 | StrictMode / Main Thread Checker |
| Scroll jank | Systrace / Instruments Time Profiler |
| 시작 느림 | App Startup tracing / Launch Time |
| Background → FG 정지 | onResume 무거움 — async 로 |
❌ 안티패턴
- runBlocking on main: 코루틴 의의 무효, 즉시 ANR.
- main 에서 큰 JSON parse: 1MB+ 면 hang.
- synchronized 가 IO 보유: 다른 스레드도 같이 멈춤.
- main 에서 SharedPreferences.commit(): synchronous disk write.
- 이미지 main decode: Glide/Coil 로 background.
- DB query 직접 Adapter 안: Paging 또는 Room Flow.
- dispatchSemaphore wait main: 죽음.
🤖 LLM 활용 힌트
- 모든 IO / decode / parse → background.
- StrictMode + Main Thread Checker 항상 dev 켜기.