0

我在我的应用程序中使用了这个示例代码,但是当我关闭相机视图控制器时出现问题,并且在动画完成之前很快,我再次单击返回相机视图控制器的按钮。

崩溃说:

libswiftCore.dylib`_swift_abortRetainUnowned:
0x83895e <+0>:  movw   r0, #0xe7ae
0x838962 <+4>:  movt   r0, #0x3
0x838966 <+8>:  add    r0, pc
0x838968 <+10>: ldr    r0, [r0]
0x83896a <+12>: movw   r1, #0xc1ad
0x83896e <+16>: movt   r1, #0x1
0x838972 <+20>: add    r1, pc
0x838974 <+22>: movs   r2, #0x0
0x838976 <+24>: str    r2, [r0, #0xc]
0x838978 <+26>: str    r1, [r0, #0x8]

-> 0x83897a <+28>:陷阱

当我试图跟踪问题时,有时会导致我想到以下几行:

DispatchQueue.main.async { [unowned self] in
            self.recordButton.isEnabled = self.movieFileOutput != nil
        }
    }

有时到以下内容:

self.movieFileOutput = movieFileOutput

任何想法?

thread #7: tid = 0x3ec7a, 0x007a097a libswiftCore.dylib`_swift_abortRetainUnowned + 28, queue = 'session queue', stop reason = EXC_BREAKPOINT (code=EXC_ARM_BREAKPOINT, subcode=0xdefe)
* frame #0: 0x007a097a libswiftCore.dylib`_swift_abortRetainUnowned + 28
frame #1: 0x007ad1bc libswiftCore.dylib`swift_unknownUnownedLoadStrong + 50
frame #2: 0x000aca0c quichar`CameraViewController.(self=0x176e5e01) -> ()).(closure #2) + 1804 at CameraViewController.swift:229
frame #3: 0x000a9a18 quichar`thunk + 56 at CameraViewController.swift:0
frame #4: 0x00e47d56 libdispatch.dylib`_dispatch_call_block_and_release + 10
frame #5: 0x00e53e62 libdispatch.dylib`_dispatch_queue_serial_drain + 980
frame #6: 0x00e4b204 libdispatch.dylib`_dispatch_queue_invoke + 556
frame #7: 0x00e54390 libdispatch.dylib`_dispatch_queue_override_invoke + 410
frame #8: 0x00e55d9e libdispatch.dylib`_dispatch_root_queue_drain + 408
frame #9: 0x00e55ba6 libdispatch.dylib`_dispatch_worker_thread3 + 112
frame #10: 0x1c49b936 libsystem_pthread.dylib`_pthread_wqthread + 1168
frame #11: 0x1c49b490 libsystem_pthread.dylib`start_wqthread + 8

整个功能如下:

override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)

    sessionQueue.async {
        switch self.setupResult {
            case .success:
                // Only setup observers and start the session running if setup succeeded.
                self.addObservers()
                self.session.startRunning()
                self.isSessionRunning = self.session.isRunning

            case .notAuthorized:
                DispatchQueue.main.async { [unowned self] in
                    let message = NSLocalizedString("AVCam doesn't have permission to use the camera, please change privacy settings", comment: "Alert message when the user has denied access to the camera")
                    let alertController = UIAlertController(title: "AVCam", message: message, preferredStyle: .alert)
                    alertController.addAction(UIAlertAction(title: NSLocalizedString("OK", comment: "Alert OK button"), style: .cancel, handler: nil))
                    alertController.addAction(UIAlertAction(title: NSLocalizedString("Settings", comment: "Alert button to open Settings"), style: .`default`, handler: { action in
                        UIApplication.shared.open(URL(string: UIApplicationOpenSettingsURLString)!, options: [:], completionHandler: nil)
                    }))

                    self.present(alertController, animated: true, completion: nil)
                }

            case .configurationFailed:
                DispatchQueue.main.async { [unowned self] in
                    let message = NSLocalizedString("Unable to capture media", comment: "Alert message when something goes wrong during capture session configuration")
                    let alertController = UIAlertController(title: "AVCam", message: message, preferredStyle: .alert)
                    alertController.addAction(UIAlertAction(title: NSLocalizedString("OK", comment: "Alert OK button"), style: .cancel, handler: nil))

                    self.present(alertController, animated: true, completion: nil)
                }
        }
    }


// Switch to Video
    sessionQueue.async { [unowned self] in
        let movieFileOutput = AVCaptureMovieFileOutput()

        if self.session.canAddOutput(movieFileOutput) {
            self.session.beginConfiguration()
            self.session.addOutput(movieFileOutput)
            self.session.sessionPreset = AVCaptureSessionPreset352x288
            if let connection = movieFileOutput.connection(withMediaType: AVMediaTypeVideo) {
                if connection.isVideoStabilizationSupported {
                    connection.preferredVideoStabilizationMode = .auto
                }

            }
            self.session.commitConfiguration()
            //the first place where the problem arise.

            self.movieFileOutput = movieFileOutput
            DispatchQueue.main.async { [unowned self] in
                self.recordButton.isEnabled = true
            }
        }
    }
//Switch Cameras
    recordButton.isEnabled = false

    sessionQueue.async { [unowned self] in
        if UIImagePickerController.isSourceTypeAvailable(.camera) {
        //Do Nothing
        } else {
            return
        }

        let currentVideoDevice = self.videoDeviceInput.device
        let currentPosition = currentVideoDevice!.position

        let preferredPosition: AVCaptureDevicePosition
        let preferredDeviceType: AVCaptureDeviceType

        switch currentPosition {
        case .unspecified, .front:
            preferredPosition = .back
            preferredDeviceType = AVCaptureDeviceType.builtInDualCamera

        case .back:
            preferredPosition = .front
            preferredDeviceType = .builtInWideAngleCamera
        }

        let devices = self.videoDeviceDiscoverySession.devices!
        var newVideoDevice: AVCaptureDevice? = nil

        // First, look for a device with both the preferred position and device type. Otherwise, look for a device with only the preferred position.
        if let device = devices.filter({ $0.position == preferredPosition && $0.deviceType == preferredDeviceType }).first {
            newVideoDevice = device
        }
        else if let device = devices.filter({ $0.position == preferredPosition }).first {
            newVideoDevice = device
        }

        if let videoDevice = newVideoDevice {
            do {
                let videoDeviceInput = try AVCaptureDeviceInput(device: videoDevice)

                self.session.beginConfiguration()

                // Remove the existing device input first, since using the front and back camera simultaneously is not supported.
                self.session.removeInput(self.videoDeviceInput)

                if self.session.canAddInput(videoDeviceInput) {
                    NotificationCenter.default.removeObserver(self, name: Notification.Name("AVCaptureDeviceSubjectAreaDidChangeNotification"), object: currentVideoDevice!)

                    NotificationCenter.default.addObserver(self, selector: #selector(self.subjectAreaDidChange), name: Notification.Name("AVCaptureDeviceSubjectAreaDidChangeNotification"), object: videoDeviceInput.device)

                    self.session.addInput(videoDeviceInput)
                    self.videoDeviceInput = videoDeviceInput
                }
                else {
                    self.session.addInput(self.videoDeviceInput);
                }

                if let connection = self.movieFileOutput?.connection(withMediaType: AVMediaTypeVideo) {
                    if connection.isVideoStabilizationSupported {
                        connection.preferredVideoStabilizationMode = .auto
                    }

                }

                /*
                 Set Live Photo capture enabled if it is supported. When changing cameras, the
                 `isLivePhotoCaptureEnabled` property of the AVCapturePhotoOutput gets set to NO when
                 a video device is disconnected from the session. After the new video device is
                 added to the session, re-enable Live Photo capture on the AVCapturePhotoOutput if it is supported.
                 */
                self.photoOutput.isLivePhotoCaptureEnabled = self.photoOutput.isLivePhotoCaptureSupported;

                self.session.commitConfiguration()
            }
            catch {
                print("Error occured while creating video device input: \(error)")
            }
        }
        //second place that is stop on

        DispatchQueue.main.async { [unowned self] in
            self.recordButton.isEnabled = self.movieFileOutput != nil
        }
    }



}
4

0 回答 0