Я схожу с ума, пытаясь заставить AVCaptureSession (в контроллере представления)быть представленным и отклоненным в моем проекте. В настоящее время я использую iOS5.1 и у меня включен ARC.
Я могу заставить его работать нормально при первом представлении контроллера представления и запуске сеанса, но когда я закрою его и представлю во второй раз, сеанс не запустится. Я подписался на уведомление «AVCaptureSessionRuntimeErrorNotification» и получаю следующую ошибку:
«Домен ошибки = Код AVFoundationErrorDomain = -11819 «Невозможно выполнить действие» UserInfo = 0x1a4020 {NSLocalizedRecoverySuggestion = Повторите попытку позже., NSLocalizedDescription = Не удается выполнить действие}
Я предполагаю, что что-то не высвобождается должным образом в моем сеансе, но с ARC нет выпусков, и вместо этого я установил все, чтобы выпуск был нулевым.
мои методы viewDidLoad просто запускают initCamera
метод initCamera:
AVCaptureSession *tmpSession = [[AVCaptureSession alloc] init];
session = tmpSession;
session.sessionPreset = AVCaptureSessionPresetMedium;
captureVideoPreviewLayer = [[AVCaptureVideoPreviewLayer alloc] initWithSession:session];
captureVideoPreviewLayer.frame = self.vImagePreview.bounds;
[self.vImagePreview.layer addSublayer:captureVideoPreviewLayer];
rearCamera = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
NSError *error = nil;
input = [AVCaptureDeviceInput deviceInputWithDevice:rearCamera error:&error];
if (!input) {
// Handle the error appropriately.
NSLog(@"ERROR: trying to open camera: %@", error);
}
[session addInput:input];
videoDataOutput = [[AVCaptureVideoDataOutput alloc] init];
NSDictionary *outputSettings = [[NSDictionary alloc] initWithObjectsAndKeys: [NSNumber numberWithInt:kCVPixelFormatType_32BGRA], kCVPixelBufferPixelFormatTypeKey, nil];
[videoDataOutput setVideoSettings:outputSettings];
[videoDataOutput setAlwaysDiscardsLateVideoFrames:YES];
queue = dispatch_queue_create("cameraQueue", DISPATCH_QUEUE_SERIAL);
[videoDataOutput setSampleBufferDelegate:self queue:queue];
dispatch_release(queue);
[session addOutput:videoDataOutput];
NSNotificationCenter *notify =
[NSNotificationCenter defaultCenter];
[notify addObserver: self
selector: @selector(onVideoError:)
name: AVCaptureSessionRuntimeErrorNotification
object: session];
[session startRunning];
[rearCamera lockForConfiguration:nil];
rearCamera.whiteBalanceMode = AVCaptureWhiteBalanceModeContinuousAutoWhiteBalance;
rearCamera.exposureMode = AVCaptureExposureModeContinuousAutoExposure;
rearCamera.focusMode = AVCaptureFocusModeContinuousAutoFocus;
[rearCamera unlockForConfiguration];
метод
captureOutput:(AVCaptureOutput *)captureOutput didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer fromConnection:(AVCaptureConnection*)соединение
вызывается без проблем при первом представлении модального контроллера представления, но при второй попытке этот метод перестает вызываться (,потому что сеанс не запускается)
Для очистки Я вызываю stopSession из моего родительского контроллера просмотра перед тем, как закрыть, и это делает следующее:
if ([session isRunning]) {
[session removeInput:input];
[session stopRunning];
[vImagePreview removeFromSuperview];
vImagePreview = nil;
input = nil;
videoDataOutput = nil;
captureVideoPreviewLayer = nil;
session = nil;
queue = nil;
}
Я чувствую, что пробовал все виды вещей, таких как выполнение отправки_синхронизации(очереди, ^{})в очереди, чтобы дождаться его сброса,но это, кажется, не имеет значения, (при вызове диспетчера_синхронизации я удалил вызов диспетчера_релиза в моем методе инициализации камеры). Я также пытался использовать метод dispatch_set_finalizer_f(queue, capture_cleanup), предложенный в другом вопросе, но я не знаю, что нужно на самом деле сделать метод очистки захвата_, потому что все примеры, которые я нахожу, не-ARC-код, где они вызывают освобождение указателя на себя. Я также просмотрел все образцы кода, которые смог найти в Apple (SquareCam и AVCam), но они тоже не-ARC. Любая помощь будет принята с благодарностью.