Нераспознанный селектор CoreData отправлен в экземпляр

#ios #swift #core-data #swiftui

#iOS #swift #core-данные #swiftui

Вопрос:

Я пытаюсь добавить CoreData в свое приложение. Это приложение будет приложением для фотографий и предоставит пользователю возможность хранить фотографии в альбомах. В настоящее время в CoreData есть две сущности, Photo и Album .

Album Объект имеет четыре атрибута. albumCoverImageData: Binary Data , id: UUID , name: String , passwordProtected: Boolean . В дополнение к этим атрибутам, он имеет photos Photo обратную связь назначения Album .

Photo Объект имеет два атрибута id: UUID и imageData: BinaryData

При попытке добавить новый альбом в базу данных при попытке сохранить контекст приложение завершает работу со следующей ошибкой.

 2020-10-20 15:49:33.889808-0400 LockIt[902:92475] -[__NSConcreteUUID compare:]: unrecognized selector sent to instance 0x2817b4fe0

2020-10-20 15:53:28.604954-0400 LockIt[902:92475] [error] error: Serious application error.  Exception was caught during Core Data change processing.  This is usually a bug within an observer of NSManagedObjectContextObjectsDidChangeNotification-[__NSConcreteUUID compare:]: unrecognized selector sent to instance 0x2817b4fe0 with userInfo (null)

CoreData: error: Serious application error.  Exception was caught during Core Data change processing.  This is usually a bug within an observer of NSManagedObjectContextObjectsDidChangeNotification-[__NSConcreteUUID compare:]: unrecognized selector sent to instance 0x2817b4fe0 with userInfo (null)

2020-10-20 15:53:28.622315-0400 LockIt[902:92475] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSConcreteUUID compare:]: unrecognized selector sent to instance 0x2817b4fe0'

*** First throw call stack:

(0x1937c65ac 0x1a78b442c 0x1936d0a2c 0x1937c9130 0x1937cb420 0x194b18c90 0x1998e5dc8 0x1998e58a0 0x1998e6ad0 0x1998e77b4 0x19989a410 0x199767b64 0x1998e7100 0x193725764 0x193725718 0x193724cd4 0x1937246a0 0x1949bc5f4 0x199892118 0x1998a0c50 0x199892fc0 0x199761aa0 0x104503bd8 0x104502e68 0x104502854 0x199fd2c04 0x19a2c3ec0 0x19a049990 0x19a0499b8 0x19a049990 0x19a03a1dc 0x19a088444 0x19a4d8f58 0x19a4d71a4 0x19a4d7d78 0x195c4d334 0x196177a4c 0x195c43350 0x195c43030 0x19612ef80 0x196108bc0 0x196190118 0x196193070 0x19618a5f4 0x19374381c 0x193743718 0x193742a28 0x19373cd20 0x19373c4bc 0x1aa24e820 0x1960e9164 0x1960ee840 0x1044f2178 0x193403e40)

libc  abi.dylib: terminating with uncaught exception of type NSException

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[__NSConcreteUUID compare:]: unrecognized selector sent to instance 0x2817b4fe0'

terminating with uncaught exception of type NSException

  

Я прикрепил соответствующие файлы ниже.
Объект фотографии в модели данных
Фото Свойства CoreDataProperties

 import Foundation

import CoreData





extension Photo {



    @nonobjc public class func fetchRequest() -> NSFetchRequest<Photo> {

        return NSFetchRequest<Photo>(entityName: "Photo")

    }



    @NSManaged public var id: UUID?

    @NSManaged public var imageData: Data?

    @NSManaged public var album: Album?



}



extension Photo : Identifiable {



    public var wrappedID: UUID {

        id!

    }

    

    public var wrappedImageData: Data {

        imageData!

    }

}
  

Объект альбома в модели данных
Альбом Свойства CoreDataProperties

 import Foundation

import CoreData





extension Album {



    @nonobjc public class func fetchRequest() -> NSFetchRequest<Album> {

        return NSFetchRequest<Album>(entityName: "Album")

    }



    @NSManaged public var albumCoverImageData: Data?

    @NSManaged public var id: UUID?

    @NSManaged public var name: String?

    @NSManaged public var passwordProtected: Bool

    @NSManaged public var photos: NSSet?

    

    public var wrappedID: UUID {

        id!

    }

    

    public var wrappedAlbumCoverImageData: Data {

        albumCoverImageData!

    }

    

    public var wrappedName: String {

        name ?? "Unamed Album"

    }

    

    public var wrappedPasswordProtected: Bool {

        passwordProtected

    }

    

    public var photoArray: [Photo] {

        let set = photos as? Set<Photo> ?? []

        

        return Array(set)

    }



}



// MARK: Generated accessors for photos

extension Album {



    @objc(addPhotosObject:)

    @NSManaged public func addToPhotos(_ value: Photo)



    @objc(removePhotosObject:)

    @NSManaged public func removeFromPhotos(_ value: Photo)



    @objc(addPhotos:)

    @NSManaged public func addToPhotos(_ values: NSSet)



    @objc(removePhotos:)

    @NSManaged public func removeFromPhotos(_ values: NSSet)



}



extension Album : Identifiable {

    

}
  

Основной вид

 import SwiftUI



struct AlbumCollectionView: View {

    @Environment(.managedObjectContext) var managedObjectContext

    @FetchRequest(entity: Photo.entity(),

                  sortDescriptors: [

                    NSSortDescriptor(keyPath: Photo.id, ascending: true)

                  ]

    ) var photos: FetchedResults<Photo>

    

    @FetchRequest(entity: Album.entity(),

                  sortDescriptors: [

                    NSSortDescriptor(keyPath: Album.name, ascending: true)

                  ]

    ) var albums: FetchedResults<Album>

    

    @State private var showingImagePicker = false

    @State private var showingPopover = false

    @State private var inputImage: UIImage?

    @State private var albumNameInput: String = ""

    

    let columns = [

        GridItem(.adaptive(minimum: 150))

    ]

    

    var body: some View {

        ScrollView {

            LazyVGrid(columns: columns, spacing: 20) {

                ForEach(albums) { album in

                    NavigationLink(destination: AlbumView(album: album)) {

                        VStack {

                            Image(data: album.albumCoverImageData, placeholder: "No Image")

                                .resizable()

                                .aspectRatio(1, contentMode: .fill)

                                .contextMenu(menuItems: {

                                    Button(action: {

                                        deleteAlbum(selectedAlbum: album)

                                    }) {

                                        Label("Remove", systemImage: "trash")

                                    }

                                })

                            Text(album.name ?? "")

                        }

                    }

                }

            }

            .padding()

            .navigationBarTitle(Text("Albums"))

            .navigationBarItems(trailing:

                Menu {

                    Button(action: {

                        self.showingPopover = true

                    }) {

                        Label("Add Album", systemImage: "photo.on.rectangle.angled")

                    }

                    Button(action: {

                        self.showingImagePicker = true

                    }) {

                        Label("Add Photo", systemImage: "photo")

                    }

                } label: {

                    Image(systemName: "plus")

                        .popover(isPresented: $showingPopover, arrowEdge: .trailing ,content: {

                            VStack {

                                Text("New Album")

                                TextField("Enter Album Name",text: $albumNameInput)

                                    .textFieldStyle(RoundedBorderTextFieldStyle())

                                Button("Submit"){

                                    addAlbum(name: albumNameInput, albumCover: UIImage(named: "Image 1")!)

                                    albumNameInput = ""

                                    showingPopover = false

                                }

                            }

                            .padding()

                        })

                }

            )

            .sheet(isPresented: $showingImagePicker, onDismiss: loadImage) {

                ImagePicker(image: self.$inputImage)

            }

        }

    }

    

    func addAlbum(name: String, albumCover: UIImage) {

        let newAlbum = Album(context: managedObjectContext)

        

        newAlbum.id = UUID()

        newAlbum.name = name

        newAlbum.passwordProtected = false

        newAlbum.albumCoverImageData = albumCover.jpegData(compressionQuality: 1.0)

        

        saveContext()

    }

    

    func deleteAlbum(selectedAlbum: Album) {

        for album in albums {

            if selectedAlbum == album {

                self.managedObjectContext.delete(album)

            }

        }

        saveContext()

    }

    

    func loadImage() {

        guard let inputImage = inputImage else { return }

        addPhoto(image: inputImage)

    }

    

    func addPhoto(image: UIImage) {

        let newPhoto = Photo(context: managedObjectContext)

        

        newPhoto.id = UUID()

        newPhoto.imageData = image.jpegData(compressionQuality: 1.0)

        

        saveContext()

    }

    

    func saveContext() {

        do {

            try managedObjectContext.save()

        } catch {

            print("Error saving managed object context: (error)")

        }

    }

}
  

Я в недоумении относительно того, что здесь может происходить, ценю любую информацию!

Комментарии:

1. Вы недавно меняли свою модель основных данных без создания новых файлов (или вручную обновляли существующие)?

2. Я в это не верю, я могу обновить сообщение скриншотом, чтобы убедиться, что различий нет, но, насколько мне известно, у меня их нет, и не должно быть никаких различий.

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

4. Я не изменил его, но я также еще не сгенерировал новые файлы. После того, как я сгенерировал файлы, я добавил wrapped свойства, но это единственные изменения, которые я внес.

5. Не по теме, но, возможно, было бы лучше добавить ваши собственные вычисляемые свойства и другие данные в расширение к классу сущностей в отдельном файле

Ответ №1:

Вы говорите, что проблема возникает при добавлении и сохранении нового Album , хотя я думаю, что проблема на самом деле связана Photos с. Ошибка:

причина: ‘-[__NSConcreteUUID compare:]: нераспознанный селектор отправлен в экземпляр

используется в старой терминологии Objective-C (CoreData по своей сути основана на Objective-C). Это указывает на то, что существует объект типа __NSConcreteUUID (который предположительно является типом объекта, который CoreData использует для хранения ваших UUID), и CoreData пытается вызвать compare метод для этого объекта. compare это метод, который вызывается при сортировке объектов: сравните объект A с объектом B, чтобы определить, в каком порядке они должны располагаться относительно друг друга.

Итак, источник проблемы заключается в том, что ваши представления сортируют что-то по UUID, и это, похоже, фотографии:

      @FetchRequest(entity: Photo.entity(),
         sortDescriptors: [
             NSSortDescriptor(keyPath: Photo.id, ascending: true)
             ]
         ) var photos: FetchedResults<Photo>
  

но идентификаторы UUID не могут быть отсортированы таким образом. Если нет веских оснований для сортировки по UUID, я бы просто изменил порядок сортировки, чтобы использовать другой атрибут.

Комментарии:

1. Спасибо! Знаете ли вы документацию, которая объясняла бы, почему UUID нельзя сортировать таким образом, и каково общее правило для того, какие ключи нельзя использовать для сортировки? Кажется естественным иметь объект с идентификатором UUID, а затем сортировать по этому идентификатору, чтобы порядок был согласованным.