[G1-Sync] Manual knowledge update

This commit is contained in:
Antigravity Agent
2026-05-09 21:08:02 +09:00
parent f0befc887a
commit 93ec7e9056
363 changed files with 68333 additions and 64 deletions
@@ -0,0 +1,94 @@
---
id: android-compose-recomposition-pitfalls
title: Compose Recomposition 함정
category: Coding
status: draft
source_trust_level: B
verification_status: conceptual
created_at: 2026-05-09
updated_at: 2026-05-09
tags: [android, compose, recomposition, performance, vibe-coding]
tech_stack: { language: "Kotlin / Jetpack Compose", applicable_to: ["Android"] }
applied_in: []
aliases: [stable, immutable, skippable, derivedStateOf]
---
# Compose Recomposition 함정
> Compose 는 "필요한 부분만 다시 그린다" 가 디폴트지만, **stable / immutable / skippable** 조건이 깨지면 매번 다시 그림. Layout Inspector + Compose Compiler Metrics 로 측정 후 최적화.
## 📖 핵심 개념
- Stable: 필드가 변하지 않거나 변경 시 알림. compose 가 ==/equals 안전 비교 가능.
- Immutable: 자체적으로 mutate 불가 (val + immutable types).
- Skippable: stable 인 매개변수만 받는 composable. 같은 args → skip.
## 💻 코드 패턴
### Immutable data class
```kotlin
@Immutable
data class User(val id: String, val name: String) // 모든 필드 val + immutable type
@Composable
fun UserCard(user: User) { ... } // skippable
```
### List 는 ImmutableList
```kotlin
import kotlinx.collections.immutable.ImmutableList
import kotlinx.collections.immutable.persistentListOf
@Composable
fun Items(items: ImmutableList<Item>) { // 일반 List 는 unstable
LazyColumn { items(items) { Row(it) } }
}
```
### derivedStateOf — 파생 값 캐시
```kotlin
val showButton by remember(items) {
derivedStateOf { items.isNotEmpty() && items.all { it.isValid } }
}
// items 변하지 않으면 계산 skip
```
### key() 로 강제 재생성
```kotlin
@Composable
fun Animated(target: T) {
key(target.id) {
AnimatedContent(target) { ... } // id 바뀌면 새 instance
}
}
```
### Lambda — remember 없이 인라인 OK (Compose 가 안정화)
```kotlin
Button(onClick = { vm.onClick() }) { ... } // OK — Compose 1.x 부터 안정
```
## 🤔 의사결정 기준
| 상황 | 도구 |
|---|---|
| 큰 list | LazyColumn / LazyVerticalGrid + key |
| 자식이 매번 재구성됨 (불필요) | 데이터 타입 immutable / stable 점검 |
| 비싼 계산 | `remember(deps) { compute() }` |
| 파생 state | `derivedStateOf` |
| 부모 state 변경에 무관한 자식 | `Modifier.composed` 또는 별도 composable |
## ❌ 안티패턴
- **일반 List<T> 를 prop 으로**: Compose 가 unstable 로 봄. Recomposition 빈발. ImmutableList.
- **Map<K,V> 일반 사용**: 마찬가지. ImmutableMap.
- **lambda 안 의 state 가 매번 다른 참조**: skippable 깨짐. 보통은 OK 지만 문제되면 remember.
- **remember 없이 비싼 객체 매번 생성**: GC 부담.
- **Modifier 가 매번 새 인스턴스**: skippable 영향. 자주 그러면 미리 만들거나 `Modifier.composed`.
- **ViewModel 의 mutableStateOf 직접 노출**: 외부 mutate 가능. State 또는 StateFlow 로 노출.
- **profiler 없이 추측 최적화**: 측정 후 optimize.
## 🤖 LLM 활용 힌트
- "데이터 클래스는 @Immutable 또는 @Stable 마크, List 는 ImmutableList" 강조.
- Compose Compiler Metrics 로 stability 점검 권장.
## 🔗 관련 문서
- [[Android_Compose_State_Hoisting]]
- [[React_Rendering_Optimization]]