112 lines
3.4 KiB
Markdown
112 lines
3.4 KiB
Markdown
---
|
|
id: ios-combine-patterns
|
|
title: Combine — 비동기 stream / 변환 / cancel
|
|
category: Coding
|
|
status: draft
|
|
source_trust_level: B
|
|
verification_status: conceptual
|
|
created_at: 2026-05-09
|
|
updated_at: 2026-05-09
|
|
tags: [ios, combine, swift, async, vibe-coding]
|
|
tech_stack: { language: "Swift / Combine", applicable_to: ["iOS 13+"] }
|
|
applied_in: []
|
|
aliases: [Publisher, Subscriber, AnyCancellable, sink]
|
|
---
|
|
|
|
# Combine — 비동기 stream
|
|
|
|
> Apple 의 reactive framework. async/await 가 더 단순하지만, Combine 은 **변환 체인 / 동시 source 결합 / 자동 cancel** 에서 강함. SwiftUI 와 자연스럽게 통합.
|
|
|
|
## 📖 핵심 개념
|
|
- Publisher: 값 stream emit.
|
|
- Subscriber: 값 받기.
|
|
- Operator: map / filter / debounce / combineLatest / merge.
|
|
- AnyCancellable: subscription handle. dispose 시 자동 cancel.
|
|
|
|
## 💻 코드 패턴
|
|
|
|
### Search debounce (typical)
|
|
```swift
|
|
import Combine
|
|
|
|
final class SearchViewModel: ObservableObject {
|
|
@Published var query: String = ""
|
|
@Published var results: [Item] = []
|
|
private var bag = Set<AnyCancellable>()
|
|
|
|
init() {
|
|
$query
|
|
.debounce(for: .milliseconds(300), scheduler: DispatchQueue.main)
|
|
.removeDuplicates()
|
|
.filter { $0.count >= 2 }
|
|
.map { q in
|
|
URLSession.shared.dataTaskPublisher(for: searchURL(q))
|
|
.map(\.data)
|
|
.decode(type: [Item].self, decoder: JSONDecoder())
|
|
.replaceError(with: [])
|
|
}
|
|
.switchToLatest() // 이전 검색 cancel
|
|
.receive(on: DispatchQueue.main)
|
|
.assign(to: &$results)
|
|
}
|
|
}
|
|
```
|
|
|
|
### combineLatest
|
|
```swift
|
|
Publishers.CombineLatest3($email, $password, $agreedToTerms)
|
|
.map { e, p, a in e.contains("@") && p.count >= 8 && a }
|
|
.assign(to: &$canSubmit)
|
|
```
|
|
|
|
### Cancel
|
|
```swift
|
|
let task = publisher.sink(receiveValue: { ... })
|
|
// 화면 disappear
|
|
task.cancel()
|
|
// 또는 bag.removeAll()
|
|
```
|
|
|
|
### async/await 와 통합
|
|
```swift
|
|
let value = try await publisher.values.first { _ in true }!
|
|
|
|
// Or convert async to publisher
|
|
extension Publisher {
|
|
func toAsync() async throws -> Output { ... }
|
|
}
|
|
```
|
|
|
|
### NotificationCenter — Combine
|
|
```swift
|
|
NotificationCenter.default.publisher(for: UIApplication.didBecomeActiveNotification)
|
|
.sink { _ in self.refresh() }
|
|
.store(in: &bag)
|
|
```
|
|
|
|
## 🤔 의사결정 기준
|
|
| 상황 | Combine | async/await |
|
|
|---|---|---|
|
|
| 한 번 호출 | ❌ | ✅ |
|
|
| Stream of values (search, websocket) | ✅ | AsyncSequence |
|
|
| 여러 source 결합 | ✅ — combineLatest | 직접 await |
|
|
| SwiftUI @Published / assign | ✅ | manual setState |
|
|
| iOS 13 호환성 | ✅ | iOS 15+ |
|
|
| Time-based (debounce, throttle) | ✅ | 직접 구현 |
|
|
|
|
## ❌ 안티패턴
|
|
- **AnyCancellable 무시**: `_ = publisher.sink(...)` — 즉시 deallocate. store(in: &bag).
|
|
- **메인 스레드에서 무거운 변환**: receive(on: DispatchQueue.global()) 으로 분기.
|
|
- **switchToLatest 안 씀**: 검색 결과 race. 이전 결과가 새 결과 덮어씀.
|
|
- **assign 후 weak self 안 함**: cycle. assign(to: &$x) 또는 sink + [weak self].
|
|
- **error 처리 누락**: .sink 의 receiveCompletion 무시 → silent failure.
|
|
- **모든 곳 Combine**: async/await 가 더 단순한 경우 많음.
|
|
|
|
## 🤖 LLM 활용 힌트
|
|
- 검색 / 스트림 / SwiftUI binding = Combine 적합.
|
|
- 한 번 호출 = async/await.
|
|
|
|
## 🔗 관련 문서
|
|
- [[iOS_Swift_Concurrency_async_await]]
|
|
- [[iOS_SwiftUI_State_Property_Wrappers]]
|