Как обрабатывать неотправляемую структуру в Async/Await?

У меня есть представление, которое позволяет пользователю выбрать изображение. Я привязываю PhotoPicker к «элементу» PhotoPickerItem. Но в функции «getImage» я получаю сообщение об ошибке: «Отправка «photoPickerItem.some» может привести к гонкам данных».

Я думаю, проблема здесь в том, что PhotoPickerItem не является отправкой. Но как мне решить эту проблему? Ждете, пока Apple разрешит Photopickeritem соответствовать Sendable?


@MainActor
struct EditableImage: View {
    
    var selectedImageUrl: Binding<String?>
    var imageType: ImageType
    
    @State var placeHolderImage: String?
    @State private var item: PhotosPickerItem?
    @State private var image: Image?
    @State private var isUploadingImage: Bool = false
    
    private func getImage(by photoPickerItem: PhotosPickerItem?) async {

        // ERROR HERE: Sending 'photoPickerItem.some' risks causing data races
        
        if let data = try? await photoPickerItem?.loadTransferable(type: Data.self) {
            if let uiImage = UIImage(data: data) {
                let croppedImage = uiImage.cropToBounds(width: 200, height: 200)
                
                await uploadImage(uiImage: croppedImage)
            }
        }
    }
    
    private func uploadImage(uiImage: UIImage) async {
        // Upload Code ....
    }
    
    var body: some View {
        PhotosPicker(selection: $item, matching: .images) {
            editableImage()
                .opacity(isUploadingImage ? 0.3 : 1)
                .overlay {
                    if !isUploadingImage {
                        // TODO
                    } else {
                        ProgressView()
                    }
                }
                .clipShape(Circle())
        }
        .disabled(isUploadingImage)
        .onChange(of: item) {
            Task {
                await getImage(by: item)
            }
        }
    }

Я пытался

@Sendable private func getImage(by photoPickerItem: PhotosPickerItem?) async { // Code }

я смотрел сеансы Apple для перехода на Swift6, но ничего не работает

Я думаю, что один из вариантов — действительно подождать, пока Apple не обновит PhotoKit до современного параллелизма или не представит новую библиотеку. Вы сможете избавиться от предупреждений, комментируя операторы import PhotoKit с помощью @preconcurrency. Это всего лишь предположение: сам я с PhotoKit не работал.

lazarevzubov 15.06.2024 12:17

я попробовал твой подход. Но это не работает. В нем говорится: «Атрибут @preconcurrency в модуле PhotosUI не имеет никакого эффекта».

Peiper 15.06.2024 16:17

Я бы посоветовал сообщить об этом в Apple и подождать.

matt 15.06.2024 16:45

Я думаю, что лучше всего отключить предупреждение до тех пор, пока Apple не сделает его доступным для отправки. Например, я использую этот подход, чтобы отключить предупреждение, а затем вызвать предупреждение после перехода на Swift 6.

Tankista 28.06.2024 11:57

Apple исправила это в Xcode 16 Beta 3.

Peiper 09.07.2024 15:49
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
2
5
123
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Я думаю, что лучше всего отключить предупреждение до тех пор, пока Apple не сделает его доступным для отправки. Например, я использую этот подход, чтобы отключить предупреждение, даже если оно имеет обратную силу, а затем активировать предупреждение после перехода на Swift 6.

#if swift(>=6.0)
#warning("Revisit @unchecked Sendable on Swift 6 version, PhotosPickerItem should be Sendable by now.")
#endif
extension PhotosPickerItem: @unchecked Sendable {}

Или, возможно, лучшим решением будет использование небезопасной неизолированной локальной переменной, например:

nonisolated(unsafe) let unsafeValue = photoPickerItem

Другие вопросы по теме

Похожие вопросы