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…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...
【服务器压力测试】本地PC电脑作为服务器运行时出现卡顿和资源紧张(Windows/Linux)
要让本地PC电脑作为服务器运行时出现卡顿和资源紧张的情况,可以通过以下几种方式模拟或触发: 1. 增加CPU负载 运行大量计算密集型任务,例如: 使用多线程循环执行复杂计算(如数学运算、加密解密等)。运行图…...
NLP学习路线图(二十三):长短期记忆网络(LSTM)
在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...
c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...
招商蛇口 | 执笔CID,启幕低密生活新境
作为中国城市生长的力量,招商蛇口以“美好生活承载者”为使命,深耕全球111座城市,以央企担当匠造时代理想人居。从深圳湾的开拓基因到西安高新CID的战略落子,招商蛇口始终与城市发展同频共振,以建筑诠释对土地与生活的…...
【笔记】WSL 中 Rust 安装与测试完整记录
#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统:Ubuntu 24.04 LTS (WSL2)架构:x86_64 (GNU/Linux)Rust 版本:rustc 1.87.0 (2025-05-09)Cargo 版本:cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...
VisualXML全新升级 | 新增数据库编辑功能
VisualXML是一个功能强大的网络总线设计工具,专注于简化汽车电子系统中复杂的网络数据设计操作。它支持多种主流总线网络格式的数据编辑(如DBC、LDF、ARXML、HEX等),并能够基于Excel表格的方式生成和转换多种数据库文件。由此&…...
【Linux】Linux安装并配置RabbitMQ
目录 1. 安装 Erlang 2. 安装 RabbitMQ 2.1.添加 RabbitMQ 仓库 2.2.安装 RabbitMQ 3.配置 3.1.启动和管理服务 4. 访问管理界面 5.安装问题 6.修改密码 7.修改端口 7.1.找到文件 7.2.修改文件 1. 安装 Erlang 由于 RabbitMQ 是用 Erlang 编写的,需要先安…...
DAY 26 函数专题1
函数定义与参数知识点回顾:1. 函数的定义2. 变量作用域:局部变量和全局变量3. 函数的参数类型:位置参数、默认参数、不定参数4. 传递参数的手段:关键词参数5 题目1:计算圆的面积 任务: 编写一…...
