前言
前段时间因为工作的需要对项目中的相机模块进行了优化,我们项目中的相机模块是基于开源库 CameraView 进行开发的。那次优化主要包括两个方面,一个是相机的启动速度,另一个是相机的拍摄的清晰度的问题。因为时间仓促,那次只是在原来的代码的基础之上进行的优化,然而那份代码本身存在一些问题,导致相机的启动速度无法进一步提升。所以,我准备自己开发一款功能完善,并且可拓展的相机库,于是 CameraX 就诞生了。
虽然去年学习了很多的 Android 的知识,但是这并没有什么骄傲的。我觉得如果一个人学习了很多的东西,但是却没有办法做出属于自己的东西,那么即使学了也跟没学一样。相比于学习能力,我更看重人的创造力。所以我也将开发一个 Android 相机库作为个人 2019 年在 Android 上面要完成的目标之一。
Android 相加开源库的现状
要使用 Android 相机实现图片拍照功能本身并不复杂,Camera1 + SurfaceView 就可以搞定。但是如果让相机能够自由拓展,就需要花费很多的功夫。我所接触的开源库包括 Google 非官方的 CameraView,以及 CameraFragment. 两个库的设计有各自的优点和缺点。
开源库 | 优点 | 缺点 |
---|---|---|
CameraView | 1.支持基本的拍照、缩放等功能;2.支持自定义图片的宽高比;3.支持多种预览布局方式; | 1.每次获取相机支持的尺寸的时候,会先将其组装到一个有序的 Set 中,这个过程会占用一定的启动时间;2.不支持拍摄视频;3.代码堆砌,结构混乱 |
CameraFragment | 1.支持拍摄照片和视频;2.代码结构清晰 | 1.不支持缩放;2.默认宽高比4:3,无法运行时修改;3.必须基于 Fragment |
以上是两个开源库的优点和缺点,而我们可以结合它们的优缺点实现一个更加完善的相机库,同时对性能的优化和用户自定义配置,我们也提供了更多的可用的接口。
CameraX 整体结构设计
虽然文章的题目是相机开发实践,但是我们并不打算介绍太多关于如何使用 Camera API 的内容,因为本项目是开源的,读者可以自行 Fork 代码进行阅读。在这里,我们只对项目中的一些关键部分的设计思路进行说明。
链接:https://www.processon.com/view/link/5c976af8e4b0d1a5b10a4049
以上是我们相机库的整体架构的设计图,这里笔者使用了 UML 建模进行基础的架构设计(当然,并非严格遵循 UML 建模的语言规则)。下面,我们介绍下项目的关键部分的设计思路。
Camera1 还是 Camera2?
了解 Android 相机 API 的同学可能知道,在 LoliPop 上面提出了 Camera2 API. 就笔者个人的实践开发的效果来看,Camera2 相机的性能确实比 Camera1 要好得多,这体现在相机对焦的速率和相机启动的速率上。当然,这和硬件也有一定的关系。Camera2 比 Camera1 使用起来确实复杂得多,但提供的可以调用的 API 也更丰富。Camera2 的另一个问题是国内的很多手机设备对 Camera2 的支持并不好。
对于这个问题,首先,我们可以根据系统的参数来判断该设备是否支持 Camera2:
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public static boolean hasCamera2(Context context) {
if (context == null) return false;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) return false;
try {
CameraManager manager = (CameraManager) context.getSystemService(Context.CAMERA_SERVICE);
assert manager != null;
String[] idList = manager.getCameraIdList();
boolean notNull = true;
if (idList.length == 0) {
notNull = false;
} else {
for (final String str : idList) {
if (str == null || str.trim().isEmpty()) {
notNull = false;
break;
}
final CameraCharacteristics characteristics = manager.getCameraCharacteristics(str);
Integer iSupportLevel = characteristics.get(CameraCharacteristics.INFO_SUPPORTED_HARDWARE_LEVEL);
if (iSupportLevel !=