Modelo CoreData del código. O "Cómo prescindir de .XCDataModel" (Parte 1)

Si por alguna razón no le gusta el proceso de crear un modelo coreData utilizando la GUI de Xcode, o si es un programador tan loco como quiero que su aplicación elimine los archivos .XCDataModel y se ejecute un poco más rápido, entonces bienvenido a cat.


Breves antecedentes


Hace aproximadamente medio año, he estado desarrollando mi propia aplicación para iOS y, hasta hace poco, el proceso de desarrollo se desarrolló sin problemas y sin problemas en el campo del almacenamiento de datos.


Pero recientemente, el modelo coreData ha crecido a tamaños que son indecentes desde mi punto de vista, y dado que soy ciego, es físicamente difícil para mí trabajar con una gran cantidad de datos semi-visuales, decidí transferir todo lo que concierne a coreData al código.


Hay muy poco material sobre el trabajo "no visual" con coreData, por lo que la mitad de mis conclusiones y desarrollos se basan en experimentos locos en Playgrounds y no menos intentos locos por buscar información faltante en Google.


En general, vamos!


Comencemos con las clases.


Si durante el trabajo estándar con coreData describimos todas las entidades y las relaciones entre ellas en un archivo separado con el modelo, entonces en nuestro caso necesitaremos clases que describan esas entidades primero (será más fácil entender con qué estamos trabajando, y Xcode ayudará, solicitando los nombres de las clases ya creadas).


Por cierto, a continuación usaremos una extraña posdata al final de cada clase, "MO" - Objeto administrado. Esto no debe confundirse con otras clases, posiblemente similares, pero no relacionadas con coreData.


Describiremos la siguiente situación:


  • Hay una empresa (descrita por la clase CompanyMO);
  • Hay empleados (descritos por la clase WorkerMO);
  • Cada empleado puede trabajar solo en una empresa (o no trabajar en absoluto);
  • Cada compañía puede tener cualquier número de empleados (incluido ningún empleado).

@objc(CompanyMO)
public class CompanyMO: NSManagedObject {
@NSManaged public var companyName: String
@NSManaged public var workers: NSSet
}

@objc(WorkerMO)
public class WorkerMO: NSManagedObject {
@NSManaged public var firstname: String
@NSManaged public var lastname: String
@NSManaged public var company: CompanyMO?
}

.


objc() — , , .


objc(CompanyMO) CompanyMO , CompanyMO Objective-C CompanyMO. objc(NameOfClass) Objective-C.


— @NSManaged .


@NSManaged , , . , ( Apple, ), @NSManaged coreData.



— , .


, , extension .


extension NSEntityDescription {
convenience init(from classType: AnyClass) {
self.init()
self.name = NSStringFromClass(classType)
self.managedObjectClassName = NSStringFromClass(classType)
}

func addProperty(_ property: NSPropertyDescription) {
self.properties.append(property)
}
}

extension NSAttributeDescription {
convenience init(name: String, ofType: NSAttributeType, isOptional: Bool = false) {
self.init()
self.name = name
self.attributeType = ofType
self.isOptional = isOptional
}
}

.


NSEntityDescription — coreData, (, ). ( , ) (managedObjectClassName), , (, — CompanyMO, WorkerMO ).


, :


let company = NSEntityDescription()
company.name = "CompanyMO"
company.managedObjectClassName = "CompanyMO"

, , NSEntityDescription. , :


let company = NSEntityDescription(from: CompanyMO.self)
let worker = NSEntityDescription(from: WorkerMO.self)

, , .



( , ), .


var model: NSManagedObjectModel {
let _model = NSManagedObjectModel()

let companyEntity = NSEntityDescription(from: CompanyMO.self)
companyEntity.addProperty(NSAttributeDescription(name: "companyName", ofType: .stringAttributeType))
//      CompanyMO.workers,         
let workerEntity = NSEntityDescription(from: WorkerMO.self)
workerEntity.addProperty(NSAttributeDescription(name: "firstname", ofType: .stringAttributeType))
workerEntity.addPropertyNSAttributeDescription((name: "lastname", ofType: .stringAttributeType))

_model.entities = [companyEntity, workerEntity]
return _model
}

.


NSManagedObjectModel entities, NSEntityDescription.


NSManagedObjectModel.entities: [NSEntityDescription]

NSEntityDescription . , , - . , , .


NSEntityDescription , companyEntity workerEntity companyEntity.properties workerEntity.properties, NSPropertyDescription.


NSPropertyDescription NSAttributeDescription ( , , ) NSRelationshipDescription ( ).


NSRelationshipDescription .


, :


  1. CompanyMO WorkerMO;
  2. , . companyEntity companyName ( workers ), workerEntity firstname lastname.
  3. entities "" .

.


.



NSPersistentStore NSPersistentStoreCoordinator NSPersistentContainer (, . ).


coreData .


, ( ) .


persistentContainer .


lazy var persistentContainer: NSPersistentContainer {
let _container = NSPersistentContainer(name: "CyrmaxModel", managedObjectModel: model)
_container.loadPersistentStores {
(description, error) in
//  ,  -
}
return _container
}

, ( ) , .


persistentContainer coreData , .



  1. , ;
  2. (- - , );
  3. ( , "?", ).


, , , , , .


, .


, .


WorkerMO, .


@objc(WorkerMO)
public class WorkerMO: NSManagedObject {
@NSManaged public var firstname: String
@NSManaged public var lastname: String

static private var _entityDescription: NSEntityDescription?
static func entityDescription() -> NSEntityDescription {
guard self._entityDescription == nil else {
return _entityDescription!
}
let des = NSEntityDescription(from: self)
des.addProperty(NSAttributeDescription(name: "firstname", ofType: .stringAttributeType))
des.addProperty(NSAttributeDescription(name: "lastname", ofType: .stringAttributeType))
self._entityDescription = des
return self._entityDescription!
}
}

CompanyMO ( ).


.


_entityDescription , ?


, NSRelationshipDescription .


, , CompanyMO.entityDescription() NSEntityDescription, .



coreData- — . , . ( , StoryBoard, ), .


- , , StoryBoard, XCDataModel.



  1. NSManagedObjectModel;
  2. NSEntityDescription;
  3. NSPropertyDescription NSAttributeDescription;
  4. Opcional, pero aún puede leer sobre NSPersistentContainer y su inicializador init (name: managedObjectModel :) .

All Articles