Cómo hacer inserciones masivas en Core Data
Libranner Santos
17 abril, 20243min de lectura
En algunos casos, es necesario insertar grandes cantidades de registros en nuestra base de datos de Core Data. Utilizar context.insert
dentro de un bucle for
es una opción que no realiza esta operación de manera eficiente.
Core Data nos proporciona las herramientas necesarias para llevar a cabo esta tarea de manera eficiente y sencilla.
Utilizando NSBatchInsert
// 1func insertBooks(_ books: [[String: Any]]) {// 2let context = persistentContainer.viewContext// 3let request = NSBatchInsertRequest(entity: Book.entity(), objects: books)// 4let result = context.execute(request)debugPrint(result)// 5try? context.save()}
Esta es la manera más simple de utilizar NSBatchInsertRequest
. Para que tenga sentido utilizar NSBatchInsertRequest
, imaginemos que estamos recibiendo un arreglo bastante grande.
Analicemos el código:
- La función acepta un array de diccionarios, los cuales deben contener los mismos nombres y tipos de datos de los atributos de nuestra entidad en Core Data.
- Obtenemos una referencia al contexto,
NSManagedObjectContext
. - Creamos una instancia de
NSBatchInsertRequest
pasando el tipo de entidad y el arreglo de objetos que queremos insertar. - Ejecutamos la operación usando
execute(_:)
. Este método retorna el resultado de la ejecución. Imprimimos el contenido deresult
para obtener más información, en caso de error. - Por último, llamamos
save
para persistir los cambios.
Insertando registros de manera eficiente
El código visto anteriormente funciona; sin embargo, podemos mejorarlo para asegurarnos de contar con un mejor rendimiento y evitar bloquear el contexto principal. Veamos una versión mejorada:
func insertBooks(_ books: [[String: Any]]) async throws {let persistentContainer = PersistenceController.shared.containerlet viewContext = persistentContainer.viewContext// 1let childContext = NSManagedObjectContext(concurrencyType: .privateQueueConcurrencyType)childContext.parent = viewContext// 2try await childContext.perform {// 3let batchInsertRequest = NSBatchInsertRequest(entity: Item.entity(), objects: books)batchInsertRequest.resultType = .statusOnlylet result = try childContext.execute(batchInsertRequest)debugPrint(result)try childContext.save()}// 5try await viewContext.perform {try viewContext.save()}}
Veamos lo que hace este código:
- Creamos
childContext
, que es un contexto hijo que utiliza concurrencia. Indicamos que es un contexto hijo deviewContext
. Esto nos permite realizar operaciones sin bloquear el contexto principal. - Utilizamos
perform
para garantizar que lo ejecutado dentro del bloque de código a continuación no bloquee el hilo principal. - Creamos nuestra operación de inserción e indicamos que para el resultado solo nos interesa el estado de la ejecución. Otras opciones que podemos utilizar son
count
yobjectIDs
, que retornan la cantidad de registros afectados o las Ids de los objetos afectado, respectivamente. - Ejecutamos la inserción masiva y persistimos los cambios en nuestro contexto hijo.
- Por último, nos aseguramos de que los cambios también se persistan en el contexto padre.
Core Data nos ofrece APIs con el nuevo modelo de concurrencia moderna (async/await), sin embargo, tambien podemos utilizar estas funcionalidades si nuestro código no usa async/await.
Con esto, tenemos un código eficiente y que garantiza un buen rendimiento.