Android 相机库CameraView源码解析 (二) : 拍照
1. 前言
这段时间,在使用 natario1/CameraView 来实现带滤镜的预览、拍照、录像功能。
由于CameraView封装的比较到位,在项目前期,的确为我们节省了不少时间。
但随着项目持续深入,对于CameraView的使用进入深水区,逐渐出现满足不了我们需求的情况。
Github中的issues中,有些BUG作者一直没有修复。
那要怎么办呢 ? 项目迫切地需要实现相关功能,只能自己硬着头皮去看它的源码,去解决这些问题。
而这篇文章是其中关于CameraView怎么进行拍照的源码解析。
以下源码解析基于CameraView 2.7.2
implementation("com.otaliastudios:cameraview:2.7.2")
为了在博客上更好的展示,本文贴出的代码进行了部分精简

拍照的入口是cameraView.takePicture(),我们从这个方法开始解析。
2. CameraEngine.takePicture
cameraView.takePicture()会调用到mCameraEngine.takePicture(),
这个PictureResult.Stub是一个参数封装类,这里重新创建了一个PictureResult.Stub并传入takePicture()方法中。
mCameraEngine是CameraEngine抽象类,实现类有Camera1Engine和Camera2Engine。
public void takePicture() {PictureResult.Stub stub = new PictureResult.Stub();mCameraEngine.takePicture(stub);
}
我们这里以Camera2为例,可以看到这里对stub参数封装类赋值了一些参数(摄像头ID、图片格式等),并调用了onTakePicture
public void takePicture(final PictureResult.Stub stub) {final boolean metering = mPictureMetering;getOrchestrator().scheduleStateful("take picture", CameraState.BIND,new Runnable() {@Overridepublic void run() {if (isTakingPicture()) return;if (mMode == Mode.VIDEO) {throw new IllegalStateException("Can't take hq pictures while in VIDEO mode");}stub.isSnapshot = false;stub.location = mLocation;stub.facing = mFacing;stub.format = mPictureFormat;onTakePicture(stub, metering);}});
}
3. onTakePicture
接着来看onTakePicture()
设置Rotation
stub.rotation = getAngles().offset(Reference.SENSOR, Reference.OUTPUT, Axis.RELATIVE_TO_SENSOR);
设置设定好拍照图片尺寸
stub.size = getPictureSize(Reference.OUTPUT);
接着调用mPictureRecorder.take(),mPictureRecorder是PictureRecorder接口,具体实现是Full2PictureRecorder,专门用来调用Camera2 API捕获图片。
mPictureRecorder = new Full2PictureRecorder(stub, this, builder,mPictureReader);
mPictureRecorder.take();
来看一下完整的重点代码
@EngineThread
@Override
protected void onTakePicture(@NonNull final PictureResult.Stub stub, boolean doMetering) {//...省略不重要代码...//设置Rotationstub.rotation = getAngles().offset(Reference.SENSOR, Reference.OUTPUT, Axis.RELATIVE_TO_SENSOR);//设置设定好拍照图片尺寸stub.size = getPictureSize(Reference.OUTPUT);//...省略不重要代码...CaptureRequest.Builder builder = mCamera.createCaptureRequest(CameraDevice.TEMPLATE_STILL_CAPTURE);applyAllParameters(builder, mRepeatingRequestBuilder);mPictureRecorder = new Full2PictureRecorder(stub, this, builder,mPictureReader);mPictureRecorder.take();
}
4. Full2PictureRecorder.take
再来看Full2PictureRecorder.take()
@Override
public void take() {mAction.start(mHolder);
}
这里调用了mAction.start(mHolder),来看一下mAction初始化
4.1 初始化BaseAction
mAction = new BaseAction() {@Overrideprotected void onStart(ActionHolder holder) { //省略了代码,这里只看结构 }@Overridepublic void onCaptureStarted(ActionHolder holder,CaptureRequest request) { //省略了代码,这里只看结构 }@Overridepublic void onCaptureCompleted(ActionHolder holder,CaptureRequest request,TotalCaptureResult result) { //省略了代码,这里只看结构 }
};
mAction是BaseAction抽象类,有onStart、onCaptureStarted、onCaptureProgressed、onCaptureCompleted等方法。mHolder是构造方法传入过来的Camera2Engine,实现了ActionHolder接口。
4.2 BaseAction.onStart
调用了mAction.start(mHolder)后,mAction和mHolder会建立关联,也就是BaseAction和Camera2Engine会建立关联,具体代码为Camera2Engine.addAction(BaseAction)将其添加到Actions列表中,并在合适的时机回调BaseAction的onCaptureStarted、onCaptureProgressed、onCaptureCompleted方法。
mAction和mHolder建立关联后,会调用onStart方法,这里是对mPictureBuilder这个建造者设置了一些值
@Override
protected void onStart(@NonNull ActionHolder holder) {super.onStart(holder);//mPictureBuilder是一个建造者,这里给建造者设置一些值mPictureBuilder.addTarget(mPictureReader.getSurface());if (mResult.format == PictureFormat.JPEG) {mPictureBuilder.set(CaptureRequest.JPEG_ORIENTATION, mResult.rotation);}mPictureBuilder.setTag(CameraDevice.TEMPLATE_STILL_CAPTURE);//应用这个建造者holder.applyBuilder(this, mPictureBuilder);
}
再来看onCaptureStarted,调用了dispatchOnShutter来回调
@Override
public void onCaptureStarted(@NonNull ActionHolder holder,@NonNull CaptureRequest request) {super.onCaptureStarted(holder, request);if (request.getTag() == (Integer) CameraDevice.TEMPLATE_STILL_CAPTURE) {dispatchOnShutter(false);setState(STATE_COMPLETED);}
}
4.3 BaseAction.onCaptureCompleted
再来看onCaptureCompleted,主要是在DNG格式的时候,做了一些特殊处理。
@Override
public void onCaptureCompleted(ActionHolder holder, CaptureRequest request, TotalCaptureResult result) {if (mResult.format == PictureFormat.DNG) {mDngCreator = new DngCreator(holder.getCharacteristics(this), result);mDngCreator.setOrientation(ExifHelper.getExifOrientation(mResult.rotation));if (mResult.location != null) {mDngCreator.setLocation(mResult.location);}}
}
结果发现这里不是重点,那么重点在哪里呢 ?
5. 设置OnImageAvailableListener监听
在Full2PictureRecorder初始化构造方法中,还有这么一句
mPictureReader.setOnImageAvailableListener(this, WorkerHandler.get().getHandler());
在Android的Camera2 API中,setOnImageAvailableListener方法用于注册一个回调监听器,以在每次图像数据可用时接收通知。
5.1 onImageAvailable回调
来看onImageAvailable回调方法,这里会调用android.media.ImageReader.acquireNextImage()来获取图像数据。
然后如果是JPEG格式,则会调用readJpegImage()方法读取图像数据
最后都会调用dispatchResult来分发数据。
@Override
public void onImageAvailable(ImageReader reader) {Image image = null;try {image = reader.acquireNextImage();switch (mResult.format) {case JPEG: readJpegImage(image); break;case DNG: readRawImage(image); break;default: throw new IllegalStateException("Unknown format: " + mResult.format);}} catch (Exception e) {mResult = null;mError = e;dispatchResult();return;} finally {if (image != null) {image.close();}}dispatchResult();
}
5.2 读取JPEG数据
我们先来看下readJpegImage()方法
private void readJpegImage(@NonNull Image image) {//从Iamge中读取数据ByteBuffer buffer = image.getPlanes()[0].getBuffer();byte[] bytes = new byte[buffer.remaining()];buffer.get(bytes);mResult.data = bytes;//根据Exif设置rotationmResult.rotation = 0;ExifInterface exif = new ExifInterface(new ByteArrayInputStream(mResult.data));int exifOrientation = exif.getAttributeInt(ExifInterface.TAG_ORIENTATION,ExifInterface.ORIENTATION_NORMAL);mResult.rotation = ExifHelper.getOrientation(exifOrientation);
}
5.3 分发回调
再来看dispatchResult,最终会调用到CameraView中的dispatchOnPictureTaken,这个方法中会遍历mListeners回调列表,调用onPictureTaken()
@Override
public void dispatchOnPictureTaken(final PictureResult.Stub stub) {mUiHandler.post(new Runnable() {@Overridepublic void run() {PictureResult result = new PictureResult(stub);for (CameraListener listener : mListeners) {listener.onPictureTaken(result);}}});
}
而mListeners什么时候被添加呢 ? CameraView中有一个addCameraListener方法,专门直接添加回调。
public void addCameraListener(CameraListener cameraListener) {mListeners.add(cameraListener);
}
5.4 设置回调
所以我们只要添加了这个回调,并实现onPictureTaken方法,就可以在onPictureTaken()中获取到拍照后的图像信息了。
binding.cameraView.addCameraListener(object : CameraListener() {override fun onPictureTaken(result: PictureResult) {super.onPictureTaken(result)//拍照回调val bitmap = BitmapFactory.decodeByteArray(result.data, 0, result.data.size)bitmap?.also {Toast.makeText(this@Test2Activity, "拍照成功", Toast.LENGTH_SHORT).show()//将Bitmap设置到ImageView上binding.img.setImageBitmap(it)val file = getNewImageFile()//保存图片到指定目录ImageUtils.save(it, file, Bitmap.CompressFormat.JPEG)}}
})
6. 其他
6.1 CameraView源码解析系列
Android 相机库CameraView源码解析 (一) : 预览-CSDN博客
Android 相机库CameraView源码解析 (二) : 拍照-CSDN博客
相关文章:
Android 相机库CameraView源码解析 (二) : 拍照
1. 前言 这段时间,在使用 natario1/CameraView 来实现带滤镜的预览、拍照、录像功能。 由于CameraView封装的比较到位,在项目前期,的确为我们节省了不少时间。 但随着项目持续深入,对于CameraView的使用进入深水区,逐…...
计算机缺少d3dx9_43.dll怎么办?5个方法快速修复d3dx9_43.dll文件
在计算机使用过程中,我们常常会遇到一些错误提示,其中之一就是“d3dx9_43.dll丢失”。这个问题可能会影响到我们的游戏体验或者软件运行。为了解决这个问题,我查阅了一些资料并尝试了多种方法。在这里,我想分享一下我对d3dx9_43.d…...
2023亚太杯数学建模C题思路分析 - 我国新能源电动汽车的发展趋势
1 赛题 问题C 我国新能源电动汽车的发展趋势 新能源汽车是指以先进技术原理、新技术、新结构的非常规汽车燃料为动力来源( 非常规汽车燃料指汽油、柴油以外的燃料),将先进技术进行汽车动力控制和驱动相结 合的汽车。新能源汽车主要包括四种类型&#x…...
c语言新龟兔赛跑
以下是一个使用C语言编写的新的龟兔赛跑游戏: #include <stdio.h>#include <stdlib.h>#include <time.h>int main() { int distance, turtle_speed, rabbit_speed, turtle_time, rabbit_time, rabbit_lead; srand(time(NULL)); // 随机数种…...
Linux驱动开发——网络设备驱动(理论篇)
目录 一、前言 二、网络层次结构 三、网络设备驱动核心数据结构和函数 一、前言 网络设备驱动是 Linux 的第三大类驱动,也是我们学习的最后一类 Linux 驱动。这里我们首先简单学习一下网络协议层次结构,然后简单讨论 Linux 内核中网络实现的层次结构。…...
simulink仿真
1)系统问题 连续系统,离散系统(采样周期问题) 系统分析问题 2)求解器问题 变步长,定步长,步长时间与采样周期问题、 3)积分器问题 连续积分,离散积分问题ÿ…...
PC端页面进去先出现加载效果
自定义指令v-loading,只需要绑定Boolean即可 v-loading“loading” <el-table :data"list" border style"width: 100%" v-loading"loading"><el-table-column align"center" label"序号" width"5…...
磁盘清理在哪里?学会这4个方法,快速清理内存!
“在使用电脑的过程中,我可能经常会保存一些文件到电脑上,这也导致电脑经常出现内存不足的情况。我想问问磁盘清理在哪里呀?我应该如何打开呢?” 随着使用电脑的时间增长,用户可能经常会遇到磁盘空间不足的情况&#x…...
Error opening terminal: xterm.”的解决方法
主要是看下面这两个变量是否设置正确 $ echo $TERM $ echo $TERMINFO 通常TERM的默认值为xterm-265color, 要查看支持的term,可以ls -al /lib/terminfo/x/ 如果TERM是xterm-265color的话,TERMINFO设置为/usr/lib/terminfo make menuconfig时提示“Err…...
C#常见的设计模式-结构型模式
引言 设计模式是软件工程中用于解决常见问题的可复用解决方案。在C#编程中,常见的设计模式具有广泛的应用。本篇博客将重点介绍C#中常见的结构型设计模式,包括适配器模式、装饰器模式、代理模式、组合模式和享元模式。 目录 引言1. 适配器模式(Adapter …...
Redis分片备库切换操作
Redis分片备库切换操作 场景描述: 分片集群: 1.ipa:5001-ipa:5002 2.ipb:5001-ipb:5002 需将两个分片备库互置完成灾备 操作步骤 准备工作 主机密码:1qaz!QAZ 获取节点信息命令 /redispath/bin/redis-cli -a password -h ip -p port red…...
二叉树:leetcode1457. 二叉树中的伪回文路径
给你一棵二叉树,每个节点的值为 1 到 9 。我们称二叉树中的一条路径是 「伪回文」的,当它满足:路径经过的所有节点值的排列中,存在一个回文序列。 请你返回从根到叶子节点的所有路径中 伪回文 路径的数目。 给定二叉树的节点数目…...
【【Linux下的Petallinux 以及其他的配置】】
Linux下的Petallinux 以及其他的配置 sudo apt-get install iproute2 gawk python3 python build-essential gcc git make net-tools libncurses5-dev tftpd zlib1g-dev libssl-dev flex bison libselinux1 gnupg wget git-core diffstat chrpath socat xterm autoconf libtoo…...
13、深度学习之神经网络
深度学习是机器学习中重要的一个学科分支,它的特点就在于需要构建多层“深度”的神经网络。 人们在探索人工智能初期,就曾设想构建一个用数学方式来表达的模型,它可以模拟人的大脑,大脑我们都知道,有很多神经元,每个神经元之间通过突触链接。 神经网络的设计就是模仿了这…...
js的数组去重方法
目录 es6数组中对象去重 1. filter()用法 2. findIndex()用法 3. 去重 其他方法: 方法二:reduce()去重 1. reduce()用法 1.1 找出字符长度最长的数组成员。 1.2 扁平化二维数组 1.3 扁平化多维数组 三、总结方案: 使用Set…...
在 Next 14 的 appRouter 模式中接入 React-Redux
在 Next 14 的 appRouter 模式中接入 React-Redux 说明 Next.js 版本升级到 14 后,相比 13 版本是一个改动很大的大版本升级,很多概念或者使用方式 13 版本都有较大的区别,因此这里记录一些学习 14 版本的 Next.js 的心得体会或者问题。因为…...
aspose-words 跳过证书验证jar
优先用 aspose-words-19.3.jar ,不需要读取license.xml,导出后直接水印,jar包最好直接放在项目resource目录下直接引用,要不下载不下来 public static String doc2pdf(String fileName, String filePath) {try {String oldFile f…...
【开题报告】基于uniapp的瑜伽学习交流小程序的设计与实现
1.选题背景 瑜伽在现代社会中越来越受到人们的关注和喜爱。它不仅可以帮助人们塑造健美的身材,还能促进身心健康、提高生活质量。然而,由于瑜伽动作的复杂性和技巧性,很多初学者在学习过程中会遇到困难和挑战。 同时,由于工作和…...
【蓝桥杯单片机】应用手势传感器(串口2)
手势传感器:串口通信,可以识别左滑、右滑、单击三种手势,输出相应的固定串口数据。 控制器:IAP15F2K61S2单片机。 引脚连接: 单片机 手势传感器 P46 -> TX P47 -> RX VCC -> 5V GND->gnd main.c 程序说明:传感器与单片机的串口2进行数据交互,这里使用的是开…...
51单片机蜂鸣器发出悦耳的声音
51单片机蜂鸣器发出悦耳的声音 1.概述 这篇文章介绍单片机控制蜂鸣器入门小实验,通过该实验掌握蜂鸣器发声的原理,控制声音发出我们想听的音乐。 2.蜂鸣器发声 2.1.硬件原理 1.蜂鸣器正极接单片机20号引脚VCC,负极接19号引脚P1.7 2.20MH…...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...
MVC 数据库
MVC 数据库 引言 在软件开发领域,Model-View-Controller(MVC)是一种流行的软件架构模式,它将应用程序分为三个核心组件:模型(Model)、视图(View)和控制器(Controller)。这种模式有助于提高代码的可维护性和可扩展性。本文将深入探讨MVC架构与数据库之间的关系,以…...
DIY|Mac 搭建 ESP-IDF 开发环境及编译小智 AI
前一阵子在百度 AI 开发者大会上,看到基于小智 AI DIY 玩具的演示,感觉有点意思,想着自己也来试试。 如果只是想烧录现成的固件,乐鑫官方除了提供了 Windows 版本的 Flash 下载工具 之外,还提供了基于网页版的 ESP LA…...
HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...
MySQL 知识小结(一)
一、my.cnf配置详解 我们知道安装MySQL有两种方式来安装咱们的MySQL数据库,分别是二进制安装编译数据库或者使用三方yum来进行安装,第三方yum的安装相对于二进制压缩包的安装更快捷,但是文件存放起来数据比较冗余,用二进制能够更好管理咱们M…...
搭建DNS域名解析服务器(正向解析资源文件)
正向解析资源文件 1)准备工作 服务端及客户端都关闭安全软件 [rootlocalhost ~]# systemctl stop firewalld [rootlocalhost ~]# setenforce 0 2)服务端安装软件:bind 1.配置yum源 [rootlocalhost ~]# cat /etc/yum.repos.d/base.repo [Base…...
STM32---外部32.768K晶振(LSE)无法起振问题
晶振是否起振主要就检查两个1、晶振与MCU是否兼容;2、晶振的负载电容是否匹配 目录 一、判断晶振与MCU是否兼容 二、判断负载电容是否匹配 1. 晶振负载电容(CL)与匹配电容(CL1、CL2)的关系 2. 如何选择 CL1 和 CL…...
Qt 事件处理中 return 的深入解析
Qt 事件处理中 return 的深入解析 在 Qt 事件处理中,return 语句的使用是另一个关键概念,它与 event->accept()/event->ignore() 密切相关但作用不同。让我们详细分析一下它们之间的关系和工作原理。 核心区别:不同层级的事件处理 方…...
WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...
人工智能 - 在Dify、Coze、n8n、FastGPT和RAGFlow之间做出技术选型
在Dify、Coze、n8n、FastGPT和RAGFlow之间做出技术选型。这些平台各有侧重,适用场景差异显著。下面我将从核心功能定位、典型应用场景、真实体验痛点、选型决策关键点进行拆解,并提供具体场景下的推荐方案。 一、核心功能定位速览 平台核心定位技术栈亮…...
