Files
2nd/10_Wiki/Topics/Coding/Native_ANR_Freeze_Debugging.md
T
2026-05-09 21:08:02 +09:00

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
native
anr
freeze
performance
vibe-coding
language applicable_to
Swift / Kotlin
iOS
Android
ANR
Application Not Responding
hang
main thread block
freeze

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 켜기.

🔗 관련 문서