Буфер вывода выходного буфера AVCaptureSession сохраняется в CoreData - PullRequest
3 голосов
/ 17 апреля 2011

Я использую AVCaptureSession для захвата кадров с камеры, используя метод setSampleBufferDelegate из класса AVCaptureVideoDataOutput. Метод делегата выглядит следующим образом. Вы можете видеть, что я конвертирую в UIImage и помещаю его в UIImageView. Я хотел бы сохранить каждый UIImage на диск и сохранить URL-адрес в новом managedObject, но я не знаю, как правильно получить managedObjectContext, так как каждый вызов порождает новый поток, используя очередь последовательной отправки. Может кто-нибудь предложить решение использовать CoreData и очереди отправки таким образом, чтобы я мог создать коллекцию изображений, которые хранятся на диске и соответствуют управляемому объекту.

- (void)captureOutput:(AVCaptureOutput *)captureOutput didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer fromConnection:(AVCaptureConnection *)connection {
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

CVImageBufferRef imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer); 
/*Lock the image buffer*/
CVPixelBufferLockBaseAddress(imageBuffer,0); 
/*Get information about the image*/
uint8_t *baseAddress = (uint8_t *)CVPixelBufferGetBaseAddress(imageBuffer); 
size_t bytesPerRow = CVPixelBufferGetBytesPerRow(imageBuffer); 
size_t width = CVPixelBufferGetWidth(imageBuffer); 
size_t height = CVPixelBufferGetHeight(imageBuffer);  

/*Create a CGImageRef from the CVImageBufferRef*/
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB(); 
CGContextRef newContext = CGBitmapContextCreate(baseAddress, width, height, 8, bytesPerRow, colorSpace, kCGBitmapByteOrder32Little | kCGImageAlphaPremultipliedFirst);
CGImageRef newImage = CGBitmapContextCreateImage(newContext); 

    /*We release some components*/
    CGContextRelease(newContext); 
    CGColorSpaceRelease(colorSpace);

/*We display the result on the image view (We need to change the orientation of the image so that the video is displayed correctly).
     Same thing as for the CALayer we are not in the main thread so ...*/
    UIImage *image= [UIImage imageWithCGImage:newImage scale:1.0 orientation:UIImageOrientationRight];

/*We relase the CGImageRef*/
CGImageRelease(newImage);

[self.imageView performSelectorOnMainThread:@selector(setImage:) withObject:image waitUntilDone:YES];

/*We unlock the  image buffer*/
CVPixelBufferUnlockBaseAddress(imageBuffer,0);

[pool drain];
}

1 Ответ

2 голосов
/ 17 апреля 2011

Рекомендованное решение - создать новый NSManagedObjectContext для каждого потока, каждый из которых указывает на один NSPersistentStoreCoordinator. Вы также можете прослушать NSManagedObjectContextDidSaveNotification, чтобы объединить изменения с контекстом основного потока (используя метко mergeChangesFromContextDidSaveNotification:).

Лично мне нравится использовать такой аксессор в центральном месте для обработки контекстов для каждого потока:

- (NSManagedObjectContext *) managedObjectContext {
    NSManagedObjectContext *context = [[[NSThread currentThread] threadDictionary] objectForKey:@"NSManagedObjectContext"];
    if (context == nil) {
        context = [[[NSManagedObjectContext alloc] init] autorelease];
        [context setPersistentStoreCoordinator:self.persistentStoreCoordinator];
        [[[NSThread currentThread] threadDictionary] setObject:context forKey:@"NSManagedObjectContext"];
    }
    return context;
}

Помните, что вы не можете передавать NSManagedObjects между потоками проще, чем вы можете передавать контексты. Вместо этого вы должны передать NSManagedObjectID (из свойства objectID объекта), а затем в целевом потоке использовать метод контекста этого потока objectWithID:, чтобы получить эквивалентный объект.

...