其他分享
首页 > 其他分享> > Camera2之CameraManager类

Camera2之CameraManager类

作者:互联网

作者:nullobject0x01
链接:https://www.jianshu.com/p/1f4021aacf6b

0x01 概览

CameraManager是一个用于检测、连接和描述相机设备的系统服务,负责管理所有的CameraDevice相机设备。可以通过调用Context.getSystemService(java.lang.String)方法来获取一个CameraManager的实例:

 

CameraManager manager=(CameraManager)getSystemService(Context.CAMERA_SERVICE);```

###0x02 内部类
- **AvailabilityCallback** 
当相机设备可用状态发生改变时,会回调这个`Callback`中相应的方法

- **TorchCallback**  
闪光灯状态监听类,当闪光灯模式发生改变时,会回调这个`Callback`中相应的方法

###0x03 公共方法
- **CameraCharacteristics getCameraCharacteristics(String cameraId)**  
用于查询id为cameraId的相机设备所支持的功能。该方法会返回一个**CameraCharacteristics**类的对象,`CamraCharacteristic`类中封装了相机设备固有的的所有功能属性。可以通过该对象,获取和设置相机的参数,如对焦方式、闪光灯设置等。

- **String[] getCameraIdList()**  
用于获取当前连接的所有相机设备的`cameraId`集合,包括可能其他camera API client正在使用的相机。注意这里的`cameraId`,对于不可移动的相机设备(比如手机自带的前后置相机),这个标识符从**0**开始,一般来说,后置相机的标识符为**"0"**,常量值为``CameraCharacteristics.LENS_FACING_FRONT``,前置相机的标识符为**"1"**,常量值为``CameraCharacteristics.LENS_FACING_BACK``。然而,对于外置的相机设备,不管是相同型号或不同型号,**它们都会具有一个自己单独的标识符。**

- **void openCamera(String cameraId,CameraDevice.StateCallback callback,Handler handler)** 
用于打开一个到具有给定`cameraId`的摄像机的连接:
  1. cameraId:可通过``getCameraIdList()``方法获取,不过需要注意的两种情况:**1.**尽管某个camera可能包含在``getCameraIdList()``方法返回的数组中,但在调用``getCameraIdList()``方法之后到调用`openCamera()`方法之前的这个时间内,该`cameraId`的设备连接被断开了;**2.**该`cameraId`的相机设备正在被一个具有更高优先级的camera API client使用:**在这两种情况下仍然可能会导致相机打开失败。**
  2. CameraDevice.StateCallback:具体获取实例的方式和该内部类的各方法说明如下:

```android
private final CameraDevice.StateCallback mStateCallback1=new 
CameraDevice.StateCallback(){ 
@Override    
public void onOpened(CameraDevice camera) {
   //当camera成功打开之后会回调该方法。此时camera已经就绪,可以开始对相机进行一系列的操作,
    //可通过调用CameraCaptureSession.createCaptureSession方法来设置第一个capture session
 }    
@Override    
public void onDisconnected(CameraDevice camera){
    //当camera不再可用,或调用CameraManager.openCamera()打开相机失败时都会
    //调用此方法.此时任何尝试调用CameraDevice方法的操作都会失败并抛出一个
    //CameraAccessException异常。
    //安全策略或权限的改变、可移动相机设备的物理断开、或者当该camera需要更高优
    //先级的camera API Client时都会导致该camera设备连接断开
    }    
@Override    
public void one rror(CameraDevice camera, int error){
    //顾名思义。当camera出错时会回调该方法
    //应在该方法调用CameraDevice.close()和做一些其他释放相机资源的操作,防止相机出错而导致一系列问题。
    }    
@Override    
public void onClosed(CameraDevice camera){
  super.onClosed(camera);
  //当调用CameraDevice.close()关闭相机设备后会回调此方法。camera为别关闭的相机设备。
  //该方法被回调执行后,任何尝试调用camera相关的操作都会失败,并且抛出一个IllegalStateException异常
    }
};
  1. handler: 用于指定调用该回调的线程。如果不指定则置null,此时系统会默认使用当前线程的looper。当需要将该回调放到后台线程处理时,可通过指定该handlerlooper来设置:

 

mBackgroundThread = new HandlerThread("CameraBackground");
mBackgroundThread.start();
mBackgroundHandler = new Handler(mBackgroundThread.getLooper());

 

  CameraManager manager = (CameraManager) activity.getSystemService(Context.CAMERA_SERVICE);
manager.registerAvailabilityCallback(new CameraManager.AvailabilityCallback(){    
@Override    
public void onCameraAvailable(String cameraId)    {        
      super.onCameraAvailable(cameraId);        
      Log.e(TAG,cameraId+" is onCameraAvailable");    
}    
@Override    
public void onCameraUnavailable(String cameraId)    {        
      super.onCameraUnavailable(cameraId);        
      Log.e(TAG,cameraId+" is onCameraUnavailable");    
    }
},mBackgroundHandler);


要注意的是,每当相机设备被任何camera API client打开时onCameraUnavailable(String cameraId)方法都会被回调,此时的cameraId应为未被打开的设备id。这刚好能解释图 (1)中日志输出的最后行:因为在这个程序中,我调用registerAvailabilityCallback注册了回调之后紧跟着调用openCamera()打开了手机的后置摄像头(cameraId为0,前置摄像头id为1)。当注册了该回调之后,一旦不再需要,应立即调用CameraManager.unregisterAvailabilityCallback(CameraManager.AvailabilityCallback callback)方法来解除回调注册。否则可能会造成一些严重的问题:该回调会一直占用资源而不能被释放,且会继续而且是独立于activity生命周期和单独的CameraManager实例地无终止地接收相机事件!

 

标签:Camera2,cameraId,CameraManager,相机,camera,CameraDevice,void
来源: https://blog.csdn.net/qiuchangyong/article/details/117301329