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 运行。
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 运行。