Это перевод статьи Concurrency Step-by-Step: Reading from Storage
Тема, которая снова и снова возникает в связи с многопоточностью (concurrency) в Swift, — это попытка «сделать компилятор довольным». Вы просто хотите, чтобы глупые ошибки исчезли. Пытаясь сделать это, вы натыкаетесь на множество вещей, таких как Sendable
или @preconcurrency
. Вы даже можете начать менять класс class
на актор actor
, и непонятно, насколько это может отличаться, но это даже то же самое количество символов. Поэтому вы просто начинаете бросаться синтаксисом в проблему. Это понятно!
Иногда это может даже работать в краткосрочной перспективе. Но этот путь обычно приводит к разочарованию и гневу. Часто он приводит к чрезвычайно сложным проектам, которые просто приводят к еще большим проблемам.
Добро пожаловать во вторую часть «Concurrency Swift шаг за шагом». Первая часть «Concurrency Swift шаг за шагом» находится здесь. Цель этих постов— проработать общую задачу, чтобы помочь сформировать реальное понимание того, что происходит. В прошлый раз мы рассматривали сетевой запрос. На этот раз мы загрузим модель из хранилища данных.
Краткие заметки
Я проигнорирую обработку ошибок, чтобы сосредоточиться на многопоточности.
Я не очень хорош в SwiftUI. Нам потребуется Xcode 16 или более поздняя версия.
Мне было очень трудно придумать пример, который был бы одновременно простым и иллюстрировал проблему. Я думаю, что локальное хранилище будет работать хорошо, но нам придется сделать его довольно надуманным. Я не думаю, что это действительно уведет нас от каких-либо идей. Но я все равно хочу это подчеркнуть, потому что идея «хранилища данных» здесь не будет похожа на SwiftData, CoreData или другие вещи, которые может использовать реальное приложение.
Кроме того, этот пост строится на темах, обсуждавшихся в предыдущем посте. Некоторые из этих вещей будет сложнее понять, если вы не знакомы с содержанием предыдущего поста.
Расставляем детали по местам
Итак, начнем с определения интерфейса нашей системы хранения данных.
class DataModel {
let name: String
init(name: String) {
self.name = name
}
}
class Store {
func loadModel(named name: String) async -> DataModel {
DataModel(name: name)
}
}
Я же говорил, что это будет надуманно!
Есть только ТИП DataModel
для хранения простого значения name
, а также Store
, который «загружает» для нас модели. Ни один из них не делает ничего полезного. Но на самом деле нас интересуют только ТИПы и их интерфейсы.
Теперь нам нужен SwiftUI View
, чтобы связать все это воедино.
struct ContentView: View {
@State private var store = Store()
@State private var name: String = "---"
var body: some View {
Text("hello \(name)!")
.task {
self.name = await store.loadModel(named: "friends").name
}
}
}
Этот фрагмент кода должен очень удобно поместиться на одном экране. Неплохо!
Отступление: система ТИПов
Я вставил небольшой комментарий выше, который заслуживает большего внимания.
Но на самом деле нас интересуют только ТИПы и их интерфейсы.
Это важно и довольно нетривиально!