在我的应用程序中,我正在使用Camera2 API在后台进行一些处理。我正在使用众所周知的Camera2Basic(尽管现在已进行了大量修改)。一切运行良好,但是存在一个明显的问题-当用户向相机请求某些东西(例如启动相机应用以拍摄照片)时,我的Camera2退出了,请参阅LOGCAT:
E/Camera: Error 2
I/RequestThread-1: Flushing all pending requests.
I/RequestQueue: Repeating capture request cancelled.
I/CameraDeviceState: Legacy camera service transitioning to state ERROR
E/RequestQueue: cancel failed: no repeating request exists for request id: 0
E/CameraDeviceState: Cannot receive result while in state: 0
W/CaptureCollector: previewProduced called with no preview request on queue!
W/MessageQueue: Handler (android.os.Handler) {32bb202} sending message to a Handler on a dead thread
java.lang.IllegalStateException: Handler (android.os.Handler) {32bb202} sending message to a Handler on a dead thread
... (irrelevant) ...
at android.os.Looper.loop(Looper.java:154)
at android.os.HandlerThread.run(HandlerThread.java:61)
E/MyApp: disconnected
D/gralloc: gralloc_lock_ycbcr success. format : 11, usage: 3, ycbcr.y: 0x66903000, .cb: 0x66afd401, .cr: 0x66afd400, .ystride: 1920 , .cstride: 1920, .chroma_step: 2
E/BufferItemConsumer: [ImageReader-1920x1080f23m1-19683-0] Failed to release buffer: Unknown error -1 (1)
E/MyApp: closed
现在...在Camera1 API中,我可以在报告E / Camera Error 2时进行拦截并采取以下行动:
mCamera.setErrorCallback(errorCallback);
但是在Camera2 API中,这种情况很有趣,因为即使我已经配置了StateCallback(请注意Log.e()对应于上述LOGCAT输出),onError()也不触发,并且onClosed()和onDisconnected( )触发得太晚(!):
private CameraDevice.StateCallback mStateCallback = new CameraDevice.StateCallback() {
@Override
public void onOpened(@NonNull CameraDevice cameraDevice) {
mCameraOpenCloseLock.release();
mCameraDevice = cameraDevice;
createCameraPreviewSession();
}
@Override
public void onDisconnected(@NonNull CameraDevice cameraDevice) {
mCameraOpenCloseLock.release();
cameraDevice.close();
mCameraDevice = null;
Log.e("MyApp", "disconnected");
}
@Override
public void onClosed(@NonNull CameraDevice cameraDevice) {
super.onClosed(cameraDevice);
Log.e("MyApp", "closed");
}
@Override
public void onError(@NonNull CameraDevice cameraDevice, int error) {
Log.e("MyApp", "Here we are "+error);
mCameraOpenCloseLock.release();
cameraDevice.close();
mCameraDevice = null;
}
};
如果摄像机被用户的请求接管,为什么onError()不会触发?如何检测到这种情况?我认为能够“利用”以下任何事件都可以解决问题:
I/CameraDeviceState: Legacy camera service transitioning to state ERROR
E/RequestQueue: cancel failed: no repeating request exists for request id: 0
E/CameraDeviceState: Cannot receive result while in state: 0
谢谢您的任何反馈。
M.
最佳答案
好的,进一步研究这个问题,我得出了一个解决方案。问题是,尽管本机支持Camera 1 API的“传统”设备与“ Camera 2 API”一起工作,但它们的行为与本机支持Camera 2 API的“有限” /“完全”设备不同。
要检测设备的类型,可以使用以下代码段:
CameraCharacteristics characteristics = manager.getCameraCharacteristics(cameraId);
Integer deviceLevel = characteristics.get(CameraCharacteristics.INFO_SUPPORTED_HARDWARE_LEVEL); // 0 - limited, 1 - full, 2 - legacy, 3 - uber full
现在,“受限”和“完全”设备将按预期方式运行,并且当摄像机不可用时,将触发StateCallback的onDisconnect()事件,因此您可以做出反应。对于“旧版”设备,至少根据我的测试,这种情况为时已晚(根据原始问题中的LOGCAT)。
解决方案似乎是根据硬件级别在Camera API之间切换,然后使用
mCamera.setErrorCallback(errorCallback);
捕获Camera 1 API设备上的错误。像这样:CameraErrorCallback errorCallback = new CameraErrorCallback();
@SuppressWarnings("deprecation")
public class CameraErrorCallback implements android.hardware.Camera.ErrorCallback {
@Override
public void onError(int error, android.hardware.Camera camera) {
// Do something.
}
}