Camera2 实现相机预览

QQ图片20210306105325.png

目录

  • 为什么要使用 Camera2
  • 重要功能

为什么要使用 Camera2

相比 Camera1 ,2 的 api 功能更全,控制的粒度更细,单论获取摄像头采集的原始数据,Camera1 获取 NV21 格式的 YUV 数据,而 Camera2 获取的数据给是 YUV420,这是因为在底层会将 NV21 做格式转换,为开发者省去一步操作

最低兼容到 Android 5.0,使用 Camera2 的实现的代码量也大大增加了,所以 Google 后面又推出了 CameraX (Jetpack)来简化 api

相机 app 大多使用 Camera2 来开发

基础功能

  • 开启摄像头
  • 获取预览

开启摄像头

  • CameraManager 系统层相机服务
  • CameraCharacteristics 摄像头参数
CameraCharacteristics.LENS_FACING :
获取摄像头方向。前置摄像头(LENS_FACING_FRONT)或 后置摄像头(LENS_FACING_BACK)
CameraCharacteristics.INFO_SUPPORTED_HARDWARE_LEVEL:
获取当前设备支持的相机特性
CameraCharacteristics.SENSOR_ORIENTATION:
获取摄像头方向
CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP:
获取StreamConfigurationMap,它是管理摄像头支持的所有输出格式和尺寸
CameraCharacteristics.FLASH_INFO_AVAILABLE:
是否支持闪光灯
CameraCharacteristics.STATISTICS_INFO_MAX_FACE_COUNT:
同时检测到人脸的数量
CameraCharacteristics.STATISTICS_INFO_AVAILABLE_FACE_DETECT_MODES:
相机支持的人脸检测模式


获取预览

  • ImageReader 获得预览数据回调
  • CaptureRequest.Builder 设置摄像头参数,如自动对焦、开启闪光灯,建立会话前设置
  • CameraCaptureSession 摄像头会话,获取数据

CameraCaptureSession

  1. capture() 获取一次,常用于单张拍照
  2. setRepeatingRequest() 是重复请求获取图像数据,常用于预览或连拍

实践代码


public class Camera2Helper {

    private final static String TAG = "Camera2Helper";
    private static Camera2Helper INSTANCE;
    private Application mGlobalContext;
    private TextureView mTextureView;
    private Size mPreviewSize;
    private CameraDevice.StateCallback mDeviceStateCallback;
    private CameraDevice mCameraDevice;
    private HandlerThread mBgThread;
    private Handler mBgHandler;
    private Point mPreviewViewSize;
    private CaptureRequest.Builder mQuestBuidler;
    private CameraCaptureSession mCaptureSession;
    private CameraCaptureSession.StateCallback mSessionCallback;
    private ImageReader mImageReader;
    private ICamera2Callback mICamera2Callback;

    private Camera2Helper(){
    }

    public static Camera2Helper getInstance() {
        if (INSTANCE == null){
            INSTANCE = new Camera2Helper();
        }
        return INSTANCE;
    }

    public synchronized void openCamera(Application applicationContext, TextureView textureView,ICamera2Callback camera2Callback) {
        mTextureView = textureView;
        mGlobalContext = applicationContext;
        mICamera2Callback = camera2Callback;
        // 获取摄像头服务
        CameraManager cameraManager = (CameraManager) applicationContext.getSystemService(Context.CAMERA_SERVICE);
        // 获取摄像头参数
        try {
            String[] cameraIds =  cameraManager.getCameraIdList();
            if (cameraIds != null){
                CameraCharacteristics cameraCharacteristics = cameraManager.getCameraCharacteristics(cameraIds[0]);
                // 摄像头支持的所有输出格式和尺寸
                StreamConfigurationMap map = cameraCharacteristics.get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP);
                // 获取摄像头方向
                int cameraDirection = cameraCharacteristics.get(CameraCharacteristics.LENS_FACING);
                if (cameraDirection == CameraMetadata.LENS_FACING_FRONT){
                    Log.e(TAG,"前置摄像头");
                }else {
                    Log.e(TAG,"后置摄像头");
                }
                // 最佳预览尺寸
                mPreviewSize = getBestSupportedSize(new ArrayList<Size>(Arrays.asList(map.getOutputSizes(SurfaceTexture.class))));
                if (ActivityCompat.checkSelfPermission(applicationContext, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
                    return;
                }
                mImageReader = ImageReader.newInstance(mPreviewSize.getWidth(),mPreviewSize.getHeight(), ImageFormat.YUV_420_888,2);
                mBgThread = new HandlerThread(TAG);
                mBgThread.start();
                mBgHandler = new Handler(mBgThread.getLooper());
                // 设置 ImageReader 回调监听
                mImageReader.setOnImageAvailableListener(new OnImageAvailableListenerImpl(),mBgHandler);
                cameraManager.openCamera(cameraIds[0], getStateCallback(), mBgHandler);
            }
        } catch (Exception e) {
            Log.e(TAG,e.getMessage());
        }
    }

    private CameraDevice.StateCallback getStateCallback() {
        mDeviceStateCallback = new CameraDevice.StateCallback() {
            @Override
            public void onOpened(@NonNull CameraDevice camera) {
                mCameraDevice = camera;
                // 此时可以创建预览 Session
                Log.e(TAG,"摄像头打开");
                preview();
            }

            @Override
            public void onDisconnected(@NonNull CameraDevice camera) {
                mCameraDevice.close();
                mCameraDevice = null;
                Log.e(TAG,"onDisconnected");
            }

            @Override
            public void onError(@NonNull CameraDevice camera, int error) {
                mCameraDevice.close();
                mCameraDevice = null;
                Log.e(TAG,"onError");
            }
        };
        return mDeviceStateCallback;
    }


    // 预览
    private void preview() {

        if (mTextureView == null){
            return;
        }

        SurfaceTexture surfaceTexture = mTextureView.getSurfaceTexture();
        surfaceTexture.setDefaultBufferSize(mPreviewSize.getWidth(),mPreviewSize.getHeight());
        Surface surface = new Surface(surfaceTexture);
        //建立会话
        try {
            mQuestBuidler = mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW);
            // 自动对焦
            mQuestBuidler.set(CaptureRequest.CONTROL_AF_MODE, CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE);
            mQuestBuidler.addTarget(surface);
            mQuestBuidler.addTarget(mImageReader.getSurface());
            mCameraDevice.createCaptureSession(Arrays.asList(surface,mImageReader.getSurface()),getSessionCallback(),mBgHandler);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    private CameraCaptureSession.StateCallback getSessionCallback() {
        mSessionCallback = new CameraCaptureSession.StateCallback() {
            @Override
            public void onConfigured(@NonNull CameraCaptureSession session) {
                //会话建立,可以开始拿摄像头的数据
                mCaptureSession = session;
                try {
                    mCaptureSession.setRepeatingRequest(mQuestBuidler.build(),new CameraCaptureSession.CaptureCallback() {
                    },mBgHandler);
                } catch (CameraAccessException e) {
                    e.printStackTrace();
                }
            }

            @Override
            public void onConfigureFailed(@NonNull CameraCaptureSession session) {

            }
        };
        return mSessionCallback;
    }


    private Size getBestSupportedSize(List<Size> sizes) {
        Point maxPreviewSize = new Point(1920, 1080);
        Point minPreviewSize = new Point(1280, 720);
        Size defaultSize = sizes.get(0);
        Size[] tempSizes = sizes.toArray(new Size[0]);
        Arrays.sort(tempSizes, new Comparator<Size>() {
            @Override
            public int compare(Size o1, Size o2) {
                if (o1.getWidth() > o2.getWidth()) {
                    return -1;
                } else if (o1.getWidth() == o2.getWidth()) {
                    return o1.getHeight() > o2.getHeight() ? -1 : 1;
                } else {
                    return 1;
                }
            }
        });
        sizes = new ArrayList<>(Arrays.asList(tempSizes));
        for (int i = sizes.size() - 1; i >= 0; i--) {
            if (maxPreviewSize != null) {
                if (sizes.get(i).getWidth() > maxPreviewSize.x || sizes.get(i).getHeight() > maxPreviewSize.y) {
                    sizes.remove(i);
                    continue;
                }
            }
            if (minPreviewSize != null) {
                if (sizes.get(i).getWidth() < minPreviewSize.x || sizes.get(i).getHeight() < minPreviewSize.y) {
                    sizes.remove(i);
                }
            }
        }
        if (sizes.size() == 0) {
            return defaultSize;
        }
        Size bestSize = sizes.get(0);
        float previewViewRatio;
        if (mPreviewViewSize != null) {
            previewViewRatio = (float) mPreviewViewSize.x / (float) mPreviewViewSize.y;
        } else {
            previewViewRatio = (float) bestSize.getWidth() / (float) bestSize.getHeight();
        }

        if (previewViewRatio > 1) {
            previewViewRatio = 1 / previewViewRatio;
        }

        for (Size s : sizes) {
            if (Math.abs((s.getHeight() / (float) s.getWidth()) - previewViewRatio) < Math.abs(bestSize.getHeight() / (float) bestSize.getWidth() - previewViewRatio)) {
                bestSize = s;
            }
        }
        return bestSize;
    }




    private class OnImageAvailableListenerImpl implements ImageReader.OnImageAvailableListener {
        private byte[] y;
        private byte[] u;
        private byte[] v;

        @Override
        public void onImageAvailable(ImageReader reader) {
            // 可以提取 Iamge 对象了
            if (mICamera2Callback == null){
                return;
            }
            Image image= reader.acquireNextImage();
            Image.Plane[] planes =  image.getPlanes();
            if (y == null) {
//                new  了一次
//                limit  是 缓冲区 所有的大小     position 起始大小
                y = new byte[planes[0].getBuffer().limit() - planes[0].getBuffer().position()];
                u = new byte[planes[1].getBuffer().limit() - planes[1].getBuffer().position()];
                v = new byte[planes[2].getBuffer().limit() - planes[2].getBuffer().position()];
            }
            if (image.getPlanes()[0].getBuffer().remaining() == y.length) {
//                分别填到 yuv

                planes[0].getBuffer().get(y);
                planes[1].getBuffer().get(u);
                planes[2].getBuffer().get(v);
//                yuv 420
            }
            mICamera2Callback.onPreview(y, u, v, mPreviewSize, planes[0].getRowStride());
            image.close();

        }
    }


}

遇到的问题

  1. 无法获取到 IamgerReader 中的 YUV 数据
 
原因: 在创建会话时,只添加了 Surface 作为一路输入 Target,未添加 ImageReader.getSurface() 作为输出

解决方案:

(CaptureRequest.Builder)mRequestBuilder.addTarget(mImageReader.getSurface());

  1. 报异常 IllegalArgumentException: CaptureRequest contains unconfigured Input/Output Surface!


    ~D`3AD$0Q7V)BPV~CBUO@T1.png

相关链接

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容