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

4.0 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-memory-profiling Native Memory Profiling — iOS / Android leak 추적 Coding draft B conceptual 2026-05-09 2026-05-09
native
memory
profiling
leak
vibe-coding
language applicable_to
Swift / Kotlin
iOS
Android
Instruments Leaks
Android Profiler
LeakCanary
heap dump

Native Memory Profiling

Mobile = 메모리 한정. Leak 한 번 = OOM crash. iOS Instruments (Leaks/Allocations), Android Studio Profiler/LeakCanary 가 표준. Closure capture 와 Listener 등록이 leak 단골.

📖 핵심 개념

  • Heap: 객체 보관.
  • Leak: 참조가 안 끊겨 GC/ARC 가 회수 못 함.
  • Retain cycle: 두 객체가 서로 strong reference.

💻 코드 패턴

iOS — Instruments

# Xcode → Product → Profile (⌘I) → Leaks 또는 Allocations
# 시나리오 재현 → Instruments 가 cycle / 누수 표시.
# Memory Graph Debugger: Xcode debug → Memory Graph 버튼.

iOS — 일반 leak 패턴

class VC: UIViewController {
    var data: DataSource!
    override func viewDidLoad() {
        // ❌ self 강한 참조 → cycle
        data.onUpdate = { items in self.tableView.reloadData() }

        // ✅
        data.onUpdate = { [weak self] items in self?.tableView.reloadData() }

        // ❌ Timer 가 self 보유
        Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { _ in self.tick() }

        // ✅
        Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { [weak self] _ in self?.tick() }
    }
}

Android — LeakCanary

// build.gradle
debugImplementation("com.squareup.leakcanary:leakcanary-android:2.13")
// 자동 leak detection.
// Activity / Fragment / View / ViewModel 가 destroy 후에도 referenced 면 알림.

Android — 일반 leak 패턴

class MainActivity : AppCompatActivity() {
    // ❌ static — Application 살아있는 한 leak
    companion object { var instance: MainActivity? = null }

    // ❌ 익명 inner class 가 outer 보유
    private val handler = Handler(Looper.getMainLooper())
    override fun onCreate(s: Bundle?) {
        super.onCreate(s)
        handler.postDelayed({ doStuff() }, 60_000) // 1분 후, Activity 가 그 사이 종료되어도 leak
    }
    // ✅ removeCallbacks in onDestroy
    override fun onDestroy() {
        handler.removeCallbacksAndMessages(null)
        super.onDestroy()
    }
}

Android Studio Profiler

View → Tool Windows → Profiler → Memory
- Heap dump: 현재 객체 list
- Allocation tracking: 시점별 allocation
- Compare 2 heaps: 사이 늘어난 객체

비트맵 / 큰 자원

// 큰 image inSampleSize 로 축소
val opts = BitmapFactory.Options().apply { inSampleSize = 4 }
val bmp = BitmapFactory.decodeFile(path, opts)

// 사용 후 명시 recycle (안 해도 GC 처리하지만 빠름)
bmp.recycle()

🤔 의사결정 기준

증상 도구
OOM crash Heap dump → 가장 큰 객체
점진적 메모리 증가 Allocation tracking + 시점 비교
화면 이동 시 leak LeakCanary (Android) / Memory Graph (iOS)
큰 이미지 / 비디오 inSampleSize / Glide / FastImage
Background 메모리 누적 LiveData/Flow stop, Coroutine cancel

안티패턴

  • measure 없이 추측: 시간 낭비.
  • profiler off 상태로 release: dev 만 보고 prod 다름.
  • strong reference 묵시 가정: closure / inner class / listener 모두 강한 참조 default.
  • Singleton 에 Activity / Fragment / View 보관: leak 보장.
  • bitmap 그대로 메모리: 1080p 이미지 = 8MB. 다수면 OOM.
  • Job / Subscription cancel 안 함: 백그라운드에서도 살아있음.

🤖 LLM 활용 힌트

  • iOS = Instruments + [weak self]. Android = LeakCanary + lifecycle.
  • 큰 자원 = sampling / pooling.

🔗 관련 문서