【Android】ImageReader导致预览卡死(Camera2) BufferQueue has been abandoned

【Android】ImageReader导致预览卡死(Camera2) BufferQueue has been abandoned

起因

尝试写一个Camera2的Demo,第一步有surface成功了,第二步打算用ImageReader接收数据以便可以处理,但在第38帧左右的时候,预览画面就会卡死,不再继续播放,但程序并没有崩溃。

原因

像这种效果不一致,但程序并没有崩溃的现象无法从日志上直接定位到在哪一行,也有可能第三方库在某些地方做了异常捕获会错误判断导致,检查日志,发现多了一行BufferQueue has been abandoned,有一个队列被销毁了,猜测可能是gc引起的。

错误代码

private val stateCallback: CameraDevice.StateCallback = object : CameraDevice.StateCallback() {
        @SuppressLint("NewApi")
        override fun onOpened(camera: CameraDevice) {
            Log.d(test, "相机开启")
            //获取请求体
            val request = camera.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW)
            //生成ImageReader
            val imageReader = ImageReader.newInstance(1920, 1048, ImageFormat.YUV_420_888, 2)
            //imageReader添加监听器
            val handlerThread = HandlerThread("test2")
            handlerThread.start()
            imageReader.setOnImageAvailableListener({ imageListener(it) }, Handler(handlerThread.looper))
            //添加ImageReader
            request.addTarget(imageReader.surface)
            //开启通道并把请求发出去
            val outputs = ArrayList<OutputConfiguration>()
            outputs.add(OutputConfiguration(imageReader.surface))
            camera.createCaptureSession(
                SessionConfiguration(SESSION_REGULAR,
                    outputs,
                    MyApplication.getContext().mainExecutor,
                    object : CameraCaptureSession.StateCallback() {
                        override fun onConfigured(session: CameraCaptureSession) {
                            //通道开启 发送请求
                            session.setRepeatingRequest(request.build(), null, null)
                        }
                        override fun onConfigureFailed(session: CameraCaptureSession) {}
                    })
            )
        }
        override fun onDisconnected(camera: CameraDevice) {
            Log.d(test, "相机断开")
        }

        override fun onError(camera: CameraDevice, error: Int) {
            Log.d(test, "相机发生错误")
        }
}

解决方法

猜测是gc引起的,那就不gc好了,学activity被内部类持有导致不能gc而内存泄漏,解决方案就是弄个变量持有就行了
解决代码

    /**
     * 这里有一个坑,如果不保持imageReader的引用,
     * imageReader会被垃圾回收。这意味着imageReader
     * 可能只接收到几十帧的数据就会被回收,一旦被回收,就会
     * 引起通道出错,导致其他surface无法获取数据,实际表现
     * 就是卡死,但由于surface跑在子线程,并不会造成程序的
     * 崩溃
     */
    private lateinit var imageReader: ImageReader

    private val stateCallback: CameraDevice.StateCallback = object : CameraDevice.StateCallback() {
        @SuppressLint("NewApi")
        override fun onOpened(camera: CameraDevice) {
            Log.d(test, "相机开启")
            //获取请求体
            val request = camera.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW)
            //生成ImageReader
            val imageReader = ImageReader.newInstance(1920, 1048, ImageFormat.YUV_420_888, 2)
            //imageReader添加监听器
            val handlerThread = HandlerThread("test2")
            handlerThread.start()
            imageReader.setOnImageAvailableListener({ imageListener(it) }, Handler(handlerThread.looper))
            //添加ImageReader
            request.addTarget(imageReader.surface)
            //开启通道并把请求发出去
            val outputs = ArrayList<OutputConfiguration>()
            outputs.add(OutputConfiguration(imageReader.surface))
            camera.createCaptureSession(
                SessionConfiguration(SESSION_REGULAR,
                    outputs,
                    MyApplication.getContext().mainExecutor,
                    object : CameraCaptureSession.StateCallback() {
                        override fun onConfigured(session: CameraCaptureSession) {
                            //通道开启 发送请求
                            session.setRepeatingRequest(request.build(), null, null)
                        }

                        override fun onConfigureFailed(session: CameraCaptureSession) {

                        }
                    })
            )
        }

        override fun onDisconnected(camera: CameraDevice) {
            Log.d(test, "相机断开")
        }

        override fun onError(camera: CameraDevice, error: Int) {
            Log.d(test, "相机发生错误")
        }
    }

最后

可能还有别的方案处理,暂时没时间解决,这个方案是有缺陷的,如果imageReader作为变量还有别的用处,也碰不到这种错误,如果是像我这种写Demo的,这里就凭空多出来一个变量,在代码里看着强迫症都犯了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当然,这里是一个简单的 Android Camera2 设置预览的示例代码: 首先,你需要在你的 AndroidManifest.xml 文件中添加相机权限: ```xml <uses-permission android:name="android.permission.CAMERA" /> ``` 然后,在你的活动或片段类中,你需要声明一些成员变量: ```java private CameraManager cameraManager; private String cameraId; private CameraDevice cameraDevice; private CameraCaptureSession cameraCaptureSession; private CaptureRequest.Builder captureRequestBuilder; private ImageReader imageReader; private SurfaceView surfaceView; ``` 接下来,在你的 onCreate 方法中初始化这些变量: ```java surfaceView = findViewById(R.id.surfaceView); cameraManager = (CameraManager) getSystemService(Context.CAMERA_SERVICE); try { cameraId = cameraManager.getCameraIdList()[0]; } catch (CameraAccessException e) { e.printStackTrace(); } ``` 然后,在 onResume 方法中打开相机并设置预览: ```java @Override protected void onResume() { super.onResume(); surfaceView.getHolder().addCallback(new SurfaceHolder.Callback() { @Override public void surfaceCreated(SurfaceHolder holder) { openCamera(); } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { // 不需要实现 } @Override public void surfaceDestroyed(SurfaceHolder holder) { // 不需要实现 } }); } ``` 在 openCamera 方法中打开相机,并设置预览会话: ```java private void openCamera() { try { cameraManager.openCamera(cameraId, new CameraDevice.StateCallback() { @Override public void onOpened(@NonNull CameraDevice camera) { cameraDevice = camera; createCameraPreviewSession(); } @Override public void onDisconnected(@NonNull CameraDevice camera) { cameraDevice.close(); cameraDevice = null; } @Override public void onError(@NonNull CameraDevice camera, int error) { cameraDevice.close(); cameraDevice = null; } }, null); } catch (CameraAccessException e) { e.printStackTrace(); } } ``` 在 createCameraPreviewSession 方法中创建预览会话,并设置预览请求: ```java private void createCameraPreviewSession() { try { Surface surface = surfaceView.getHolder().getSurface(); captureRequestBuilder = cameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW); captureRequestBuilder.addTarget(surface); cameraDevice.createCaptureSession(Arrays.asList(surface, imageReader.getSurface()), new CameraCaptureSession.StateCallback() { @Override public void onConfigured(@NonNull CameraCaptureSession session) { cameraCaptureSession = session; try { cameraCaptureSession.setRepeatingRequest(captureRequestBuilder.build(), null, null); } catch (CameraAccessException e) { e.printStackTrace(); } } @Override public void onConfigureFailed(@NonNull CameraCaptureSession session) { // 处理配置失败的情况 } }, null); } catch (CameraAccessException e) { e.printStackTrace(); } } ``` 最后,在 onPause 方法中释放相机资源: ```java @Override protected void onPause() { super.onPause(); if (cameraCaptureSession != null) { cameraCaptureSession.close(); cameraCaptureSession = null; } if (cameraDevice != null) { cameraDevice.close(); cameraDevice = null; } } ``` 以上就是一个简单的 Android Camera2 设置预览的示例代码。希望对你有所帮助!如果还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值