[G1-Sync] Manual knowledge update
This commit is contained in:
@@ -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]]
|
||||
Reference in New Issue
Block a user