Camera2 ImageReader 在显示 "Failed to release buffer" 消息后挂起

Camera2 ImageReader hangs after a while with "Failed to release buffer" message

android 的 camera2 API 有问题。 我的最终目标是拥有一个字节数组,我可以使用 opencv 对其进行编辑,同时向用户显示预览(例如带有预览的 OCR)。 我创建了一个捕获请求并添加了一个 ImageReader 作为目标。然后在 OnImageAvailableListener 上,我获取图像,将其转换为位图,然后将其显示在 ImageView 上(并旋转它)。

我的问题是几秒钟后,预览停止(在逐渐变慢之后)并且在日志中出现以下错误:E/BufferItemConsumer:[ImageReader-1225x1057f100m2-18869 -0]释放缓冲区失败:未知错误-1(1)

如您在我的代码中所见,我已经尝试在从中获取 byte[] 后关闭 img。 我也试过清除缓冲区。 我试过关闭 ImageReader 但这当然阻止了我获取任何进一步的图像(抛出异常)。

谁能帮我理解我做错了什么?我一直在搜索 google 无济于事。

这是我的 OnImageAvailableListener,如果您需要我的更多代码来协助,请告诉我:

private final ImageReader.OnImageAvailableListener mOnImageAvailableListener
            = new ImageReader.OnImageAvailableListener() {
        @Override
        public void onImageAvailable(ImageReader reader) {

            Image img = reader.acquireLatestImage();
            final ImageView iv = findViewById(R.id.camPrev);

            try{

                if (img==null) throw new NullPointerException("null img");
                ByteBuffer buffer = img.getPlanes()[0].getBuffer();
                byte[] data = new byte[buffer.remaining()];
                buffer.get(data);
                final Bitmap b = BitmapFactory.decodeByteArray(data, 0, data.length);
                runOnUiThread(new Runnable() {
                    @Override
                    public void run() {
                        iv.setImageBitmap(b);
                        iv.setRotation(90);
                    }
                });
            } catch (NullPointerException ex){
                showToast("img is null");
            }finally {
                if(img!=null)
                    img.close();
            }
        }
    };

编辑 - 添加 cameraStateCallback

private CameraDevice.StateCallback mCameraDeviceStateCallback = new CameraDevice.StateCallback() {
        @Override
        public void onOpened(CameraDevice cameraDevice) {
            mCameraDevice = cameraDevice;
            showToast("Connected to camera!");
            createCameraPreviewSession();
        }

        @Override
        public void onDisconnected(CameraDevice cameraDevice) {
            closeCamera();
        }

        @Override
        public void onError(CameraDevice cameraDevice, int i) {
            closeCamera();
        }
    };

    private void closeCamera() {
        if (mCameraDevice != null) {
            mCameraDevice.close();
            mCameraDevice = null;
        }

    }

您似乎已将 setRepeatingRequest() 用于 Jpeg 格式。您的设备可能不完全支持此功能,还取决于您选择的图像分辨率。通常,我们在这些情况下使用 createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW),并从 ImageReader 获取 YUV 或原始格式。

我会尝试为 Jpeg 选择低分辨率:也许这足以保留 ImageReader 运行。