Как работают @Observable / Observation framework и чем он отличается от ObservableObject?
@Observable (iOS 17) отслеживает конкретные свойства через макрос, не требует @Published и Combine, перерисовывает только зависимые вью. ObservableObject публикует единый сигнал на весь объект при любом изменении.
Observation framework и @Observable
Начиная с iOS 17 и Swift 5.9, Apple представила макрос @Observable из модуля Observation. Он заменяет протокол ObservableObject и устраняет его главные недостатки. Компилятор разворачивает макрос в код, который отслеживает чтение конкретных свойств и уведомляет только те представления, которые реально обращались к этим свойствам.
Как работает @Observable
При объявлении класса с @Observable компилятор автоматически добавляет геттеры и сеттеры с вызовами _$observationRegistrar. Это позволяет системе точно фиксировать зависимости: если представление читает только model.title, а изменяется model.count, перерисовки не происходит.
import Observation
@Observable
final class CounterModel {
var count: Int = 0
var title: String = "Counter"
}
struct CounterView: View {
// Не нужен @StateObject или @ObservedObject
@State private var model = CounterModel()
var body: some View {
VStack {
Text(model.title)
Text("\(model.count)")
Button("+") { model.count += 1 }
}
}
}
Отличия от ObservableObject
С протоколом ObservableObject модель обязательно должна быть классом, а каждое свойство помечается @Published. Любое изменение любого @Published-свойства вызывает objectWillChange, что перерисовывает все подписанные представления целиком.
// Старый подход — ObservableObject
import Combine
final class LegacyCounterModel: ObservableObject {
@Published var count: Int = 0
@Published var title: String = "Counter"
}
struct LegacyCounterView: View {
@StateObject private var model = LegacyCounterModel()
// При изменении count перерисовывается ВСЕ представление,
// даже если оно читает только title
var body: some View {
Text(model.title)
}
}
Ключевые отличия:
- Гранулярность:
@Observableотслеживает конкретные свойства,ObservableObjectпубликует единый сигнал на весь объект. - Аннотации: с
@Observableне нужен@Published— все хранимые свойства отслеживаются автоматически. Чтобы исключить свойство, используется@ObservationIgnored. - Зависимости от Combine:
ObservableObjectиспользуетCombine,@Observable— нет. - Передача через тело представления: с
@Observableможно передавать объект как обычный аргумент — SwiftUI сам определит зависимости. СObservableObjectнужен@EnvironmentObjectили@ObservedObject. - Environment: с
@Observableиспользуется.environment(model)и@Environment(CounterModel.self)вместо.environmentObjectи@EnvironmentObject.
// Передача через environment с @Observable
struct RootView: View {
@State private var model = CounterModel()
var body: some View {
ContentView()
.environment(model)
}
}
struct ContentView: View {
@Environment(CounterModel.self) private var model
var body: some View {
Text("\(model.count)")
}
}
Подводные камни
@Observableработает только с классами; структуры и перечисления не поддерживаются.- При смешивании
ObservableObjectи@Observableв одном проекте легко перепутать@EnvironmentObjectи@Environment(T.self)— это приводит к краш-ошибке в рантайме. - Свойства с
@ObservationIgnoredне вызывают перерисовку — если забыть эту аннотацию на кэше или вспомогательном поле, появятся лишние ре-рендеры. - Computed properties не отслеживаются автоматически; если они зависят от отслеживаемых свойств, нужно обращаться к ним напрямую в теле
body. - Макрос требует Swift 5.9+ и iOS 17+; для поддержки iOS 16 и ниже всё равно нужен
ObservableObject. - Использование
@Bindableвместо@Bindingпри работе с@Observableобъектом — частая ошибка у тех, кто переходит со старого API. - Наблюдение не работает через границы actor isolation без дополнительной осторожности — мутации должны происходить на главном акторе.
Common mistakes
- Отвечать определением без production-сценария.
- Не называть runtime boundary, security boundary или failure mode.
- Игнорировать версию API, observability и тестовую проверку.
What the interviewer is testing
- Объясняет механизм своими словами и без выдуманных API.
- Называет реальные риски, диагностику и критерий корректности.
- Связывает ответ с текущей документацией и миграционными ограничениями.