Я использую .refreshable()
на List
в SwiftUI. С помощью следующего кода я выполняю обновление в симуляторе iOS 17.4, появляется счетчик, а затем исчезает через пару секунд.
var body: some View {
List() {
// items ...
}
.refreshable { try? await Task.sleep(nanoseconds: 3_000_000_000) }
}
Однако, когда я помещаю свою фактическую логику обновления в функцию refreshable
, счетчик больше не исчезает, хотя моя функция обновления завершается без ошибок (что я проверял с помощью точек останова в Xcode). Вместо этого спиннер просто вращается вечно.
var body: some View {
List() {
// items ...
}
.refreshable { await query.refetchAsync() }
}
query.refetchAsync()
запускает повторную выборку, которая выполняется в фоновом режиме Task
, а затем ожидает обновления, перебирая объект NotificationCenter.notifications(), возвращаемый self.client.queryUpdatesAsync()
:
public func refetchAsync() async {
self.refetch()
for await update in self.client.queryUpdatesAsync() {
if update.key != self.key { continue }
switch self.dataStatus {
case .idle, .pending: continue
default: break
}
}
}
Как исправить эту ошибку?
Интересно, как вы проверили, что «моя функция обновления завершается без ошибок». Если вы установили точку останова на default: break
, то ее срабатывание на самом деле не подтверждает завершение refetchAsync
. Попробуйте поставить print("Foo")
после await query.refetchAsync()
и поставить там точку останова.
Есть ли default: break
для разрыва цикла или для того, чтобы значение по умолчанию не было закодировано?
@Подметальщик, ты прав. Я забыл, что break
относится к switch
в Swift. почему это существует, если case
все равно не может провалиться неявно?
ваша асинхронная функция должна возвращать результаты, которые вы установили для состояния
break
в вашем switch self.dataStatus {
ломает переключатель, а не цикл for. Чтобы это исправить, вы можете использовать помеченные операторы:
public func refetchAsync() async {
self.refetch()
forLoop: for await update in self.client.queryUpdatesAsync() {
if update.key != self.key { continue }
switch self.dataStatus {
case .idle, .pending: continue
default: break forLoop
}
}
}
Единственная возможная причина этого — ваша функция refetchAsync()
никогда не завершается. Вам нужно сначала это исправить.
Поскольку мы не знаем ваш правильный вариант использования, предполагаем, что в вашем случае default
вы хотите завершить сам цикл.
Если это так, то должно работать:
public func refetchAsync() async {
self.refetch()
for await update in self.client.queryUpdatesAsync() {
if update.key != self.key { continue }
switch self.dataStatus {
case .idle, .pending: continue
default: return
}
}
}
Если это не ваш случай пользователя, то asyncSequence self.client.queryUpdatesAsync()
никогда не заканчивается из-за какой-то другой ошибки в коде. Исправьте и это.
notifications(named:object:)
возвращает асинхронную последовательность, которая никогда не заканчивается, поэтому ваш циклfor
никогда не заканчивается. Я предполагаю, что вы хотите завершить цикл в случаеdefault:
?break
здесь не разрывает цикл, а только разрываетswitch
.