1、OpenCV 简介
OpenCV(Open Source Computer Vision Library)是一个基于 BSD 许可开源发行的跨平台的计算机视觉库。可用于开发实时的图像处理、计算机视觉以及模式识别程序。由英特尔公司发起并参与开发,以 BSD 许可证授权发行,可以在商业和研究领域中免费使用。英特尔公司的 IPP 可以对 OpenCV 进行加速处理。
OpenCV 拥有 C++,Python 和 Java 接口,并且支持 Windows, Linux, Mac OS, iOS 和 Android 系统。OpenCV 实现了图像处理和计算机视觉方面的很多通用算法,它具有如下模块:
模块 | 功能 |
---|---|
Core | 核心基础模块,定义了被所有其他模块和基本数据结构(包括重要的多维数组 Mat)使用的基本函数、底层数据结构和算法函数 |
Imgproc | 图像处理模块,包括:滤波、高斯模糊、形态学处理、几何变换、颜色空间转换及直方图计算等 |
Highgui | 高层用户交互模块,包括:GUI、图像与视频 I/O 等 |
Video | 视频分析,运动分析及目标跟踪 |
Calib3d | 3D 模块,包括:摄像机标定、立体匹配、3D 重建等 |
Features2d | 二维特征检测与描述模块,包括:图像特征检测、描述、匹配等 |
Objdetect | 目标检测模块,如:人脸检测等 |
MI | 机器学习模块,包括:支持向量机、神经网络等 |
Flann | 最近邻开源库。包含一系列查找算法,自动选取最快算法的机制 |
Imgcodecs | 图像编解码模块,图像文件的读写操作 |
Photo | 图像计算(处理)模块,图像修复及去噪 |
Shape | 形状匹配算法模块。描述形状、比较形状 |
Stitching | 图像拼接 |
Superres | 超分辨率模块 |
Videoio | 视频读写模块,视频文件包括摄像头的输入 |
Videostab | 解决拍摄的视频稳定 |
Dnn | 深度神经网络 |
contrib | 可以引入额外模块 |
OpenCV 常用链接:
- OpenCV 官网
- OpenCV GitHub 主页
- OpenCV 中国 Wiki 论坛
2、OpenCV 与人工智能
计算机内部都是 0101,它是如何像人类那样认识判定事务的。简单来讲是通过模式匹配定位关键点。比如找人的眼睛,它的二进制排列方式是与其他事务不同的,将这种近似于眼睛的排列方式进行匹配即可定位到人的眼睛。此外,OpenCV 还可以用于图像建模、人工智能。
人类对于事务的认识源于从小到大的耳濡目染,计算机也是一样的,需要通过大量样本与素材的学习与训练,后续碰到未知元素时,将其套入到已有的模型中,从而进行判断与认知,这就是学习的过程,人工智能的本质就是学习。
线性回归
是人工智能的基础,也是 OpenCV 分类算法最重要的一个章节。通过一个例子解释,X1 表示业绩,X2 表示工作表现,二者被称为特征。Y 轴表示工资,也就是目标。很容易能得出 X 与 Y 的关系是:
- 业绩越高、工作表现越好,工资越高(Y 随着 X1 和 X2 的升高而升高)
- 业绩越低、工作表现越差,工资越低(Y 随着 X1 和 X2 的降低而降低)
图像的特征(Feature)是图像上最具代表性的一些点。所谓最具代表性,就是说这些点包含了图像表述的大部分信息。即使旋转、缩放,甚至调整图像亮度,这些点仍然稳定地存在,不会丢失。找出这些点,就相当于确定了这张图像,它们可以用于匹配、识别等有意义的工作。
通常来说,角点容易成为特征点。角点是指图案中处于角落的点,这些点附近像素值变化剧烈,因而容易被检测出。
通过众多样本数据可以拟合一个平面(所有点到平面距离的方差最短):
工资的发放情况会符合高斯分布:
人工智能的雏形就是高斯分布与拟合平面两个公式:
似然函数的数学模式是一个凹函数,从每个点找到最低点的过程就是学习的过程。
在机器学习中,支持向量机是在分类与回归分析中分析数据的监督式学习模型与相关的学习算法。假设我们要进行一个图像分类任务,当前有一系列标签:汽车、狗、猫、飞机……
一辆汽车的图像在计算机中可以表现为一个三维数组:
机器学习需要数据,让机器能够准确的识别一辆车,需要大概 5 万条数据。
识别一辆汽车,需先进行线性分类:
将一个图像分成多个网格,每个网格有权重和评分,将所有网格的评分乘以权重后累加,哪个事务得分值越高,是该种事务的可能性就越大:
学习率:
3、OpenCV 的配置
本节会介绍 OpenCV 在不同 IDE 中的通用配置。虽然是以 Android 为主的文章,但是由于在 Windows 的 IDE 上通常可以更快的看到图像的处理效果(比如 Windows 启动摄像头只需要调用一个方法,但是 Android 就要用 Camera、Camera2 或者 CameraX 多写一些代码才可以,并且还需要经过打包 -> 编译 -> 安装这套流程才能验证代码效果),因此有时我们会将 OpenCV 处理图像的代码先在 Windows 上实现证明效果无误后,再移植到 Android 上。所以我们也会介绍在 Windows IDE 中如何配置与开发。
官网找到想要使用的版本,比如 4.1.0,下载 Windows 和 Android 两个版本的 SDK:
3.1 Visual Studio 配置
我们首先打开 OpenCV Windows 的 exe 执行文件,它不会安装 OpenCV 的程序,而是将其解压到指定目录中。解压后的 build 目录下有多个目录,其中 include 头文件所在目录,lib 是库文件目录:
我们使用 Visual Studio 创建一个 CMake 项目,在次级的 CMakeLists.txt 中导入头文件和库文件目录,并让动态库文件 opencv_world410d.dll 链接到目标库中:
cmake_minimum_required (VERSION 3.8)
# 导入头文件目录,注意路径用斜杠隔开而不是反斜杠
include_directories("G:/Tools/OpenCV/build/include")
# 导入库文件目录
link_directories("G:/Tools/OpenCV/build/x64/vc15/lib")
# 将源代码添加到此项目的可执行文件。
add_executable (CMakeProject1 "CMakeProject1.cpp" "CMakeProject1.h")
# 将 lib 下的动态库文件 opencv_world410d 链接到目标库
target_link_libraries(CMakeProject1 opencv_world410d)
这里点击运行应该可以成功编译,并在项目的 /out/build/x64-Debug/CMakeProject1 目录下生成可执行文件:
为了项目能正常使用 OpenCV 的函数,将动态库 opencv_world410d.dll 拷贝到该目录下(或者将该文件添加到环境变量中也可以),然后在源文件中开始使用 OpenCV 展示一张图片:
#include "CMakeProject1.h"
#include "opencv2/opencv.hpp"
using namespace std;
using namespace cv;
void test1()
{
// 读取一张图片
Mat img = imread("C:/Users/69129/Desktop/bg.png");
// 显示图片
imshow("img", img);
// 将图片进行灰度处理
cvtColor(img, img, COLOR_BGR2GRAY);
// 在 img2 窗口中显示灰度处理后的图片
imshow("img2", img);
// 等待键入,避免程序直接运行结束使得图片一闪而过
waitKey();
}
int main()
{
test1();
return 0;
}
也可以使用 OpenCV 驱动摄像头进行录制:
void test2()
{
// 打开摄像头
VideoCapture capture(0);
if (!capture.isOpened()) {
cout << "打开摄像头失败!" << endl;
return;
}
Mat p;
while (true)
{
// 捕捉的图像传入矩阵 p
capture >> p;
if (p.empty()) {
cout << "读取摄像头数据失败!" << endl;
return;
}
imshow("img", p);
// ESC 退出
if (waitKey(30) == 27) {
break;
}
}
}
int main()
{
test2();
return 0;
}
3.2 CLion 配置
CLion 的配置要比同样是 Windows 平台的 Visual Studio 要复杂一些。因为 OpenCV 官方提供的动态库是 Visual Studio 编译的版本,CLion 不能直接用,只能自己通过编译 OpenCV 的源码获取动态库。
编译 OpenCV 源码需要准备以下工具:
- 编译工具:MinGW 或者 TDM-GCC,安装后将 bin 目录添加到环境变量中(注意路径中不要有空格)
- 构建工具:CMake
- 源码:Windows 版本的 SDK 的 source 目录内是源码
在 CMake 安装目录的 bin 目录下找到 cmake-gui.exe 并打开:
打开后执行如下三步:
点击 Configure 后要配置生成器:
一般情况下选择第一项,使用默认本地编译器即可。我们这里选择第二项手动指定,点击 Next 会让你指定 C 和 C++ 编译器:
我们这里使用 TDM-GCC,C 的编译器指定为 gcc,C++ 的编译器指定为 g++。点击 Finish 后开始配置,如果看到 Configuring done 字样表示配置成功。
在开始编译之前,先在 search 框内依次输入 test、java、python 并取消搜索结果中已经被勾选的项目:
因为我们使用的是 C++ 接口,用不到这些项目,取消勾选可以缩短编译时间。
生成成功会在 mingw_build 目录下生成如下文件:
在该目录中打开 cmd 执行编译命令:
# -j 表示启动几个线程进行编译
mingw32-make -j8
编译过程中可能会遇到如下错误:
-
编译 directx.cpp 时报错:
modules\core\CMakeFiles\opencv_core.dir\build.make:417: recipe for target 'modules/core/CMakeFiles/opencv_core.dir/src/directx.cpp.obj' failed mingw32-make[2]: *** [modules/core/CMakeFiles/opencv_core.dir/src/directx.cpp.obj] Error 1 mingw32-make[2]: *** Waiting for unfinished jobs.... CMakeFiles\Makefile2:1518: recipe for target 'modules/core/CMakeFiles/opencv_core.dir/all' failed mingw32-make[1]: *** [modules/core/CMakeFiles/opencv_core.dir/all] Error 2 Makefile:164: recipe for target 'all' failed mingw32-make: *** [all] Error 2
解决方法,在生成 CMake 文件之前,反选所有 OpenCL 相关选项重新生成:
-
找不到 windres.exe 工具:
[ 57%] Building RC object modules/core/CMakeFiles/opencv_core.dir/vs_version.rc.obj 'D:\Program' 不是内部或外部命令,也不是可运行的程序 或批处理文件。 D:\Program Files\TDM-GCC-64\bin\windres.exe: preprocessing failed. modules\core\CMakeFiles\opencv_core.dir\build.make:1510: recipe for target 'modules/core/CMakeFiles/opencv_core.dir/vs_version.rc.obj' failed mingw32-make[2]: *** [modules/core/CMakeFiles/opencv_core.dir/vs_version.rc.obj] Error 1 mingw32-make[2]: *** Waiting for unfinished jobs.... CMakeFiles\Makefile2:1518: recipe for target 'modules/core/CMakeFiles/opencv_core.dir/all' failed mingw32-make[1]: *** [modules/core/CMakeFiles/opencv_core.dir/all] Error 2 Makefile:164: recipe for target 'all' failed mingw32-make: *** [all] Error 2
这是由于我们配置的编译工具路径中有空格导致没有找到 windres.exe,需要将 TDM-GCC 安装到一个没有空格的路径中重新添加环境变量,重新在 CMake 中进行配置、生成、编译过程。这也是为什么我们前面提到编译工具安装路径不能有空格的原因
如果编译成功,再执行安装命令:
mingw32-make install
会在 mingw_build 目录下再生成一个 install 目录,内容如下:
最后在 CLion 的 CMakeLists.txt 文件中添加如下配置:
# 创建 OpenCV_DIR 变量
set(OpenCV_DIR G:/Tools/opencv410/mingw_build/install)
# 查找并加载 OpenCV_DIR 地址下的 OpenCVConfig.cmake 文件
find_package(OpenCV REQUIRED)
# 包含头文件目录
include_directories(${OpenCV_INCLUDE_DIRS})
# 将库文件链接到项目中
target_link_libraries(OpenCV_CLion ${OpenCV_LIBS})
这个配置方式在 /mingw_build/install/OpenCVConfig.cmake 中有所介绍:
# Usage from an external project:
# In your CMakeLists.txt, add these lines:
#
# FIND_PACKAGE(OpenCV REQUIRED)
# TARGET_LINK_LIBRARIES(MY_TARGET_NAME ${OpenCV_LIBS})
#
# Or you can search for specific OpenCV modules:
#
# FIND_PACKAGE(OpenCV REQUIRED core imgcodecs)
#
# If the module is found then OPENCV_<MODULE>_FOUND is set to TRUE.
#
# This file will define the following variables:
# - OpenCV_LIBS : The list of libraries to link against.
# - OpenCV_INCLUDE_DIRS : The OpenCV include directories.
# - OpenCV_COMPUTE_CAPABILITIES : The version of compute capability
# - OpenCV_VERSION : The version of this OpenCV build: "4.1.0"
# - OpenCV_VERSION_MAJOR : Major version part of OpenCV_VERSION: "4"
# - OpenCV_VERSION_MINOR : Minor version part of OpenCV_VERSION: "1"
# - OpenCV_VERSION_PATCH : Patch version part of OpenCV_VERSION: "0"
# - OpenCV_VERSION_STATUS : Development status of this build: ""
FIND_PACKAGE() 指示要寻找 OpenCV,REQUIRED 用于确保在找不到指定的包时发生错误并停止构建过程。
包含头文件使用的 ${OpenCV_INCLUDE_DIRS} 和链接库文件时所使用的 ${OpenCV_LIBS} 都是这个 CMake 文件生成的变量,用 message 打印变量值可以发现:
-- G:/Tools/opencv410/mingw_build/install/include
-- opencv_calib3d;opencv_core;opencv_dnn;opencv_features2d;opencv_flann;opencv_gapi;opencv_highgui;opencv_imgcodecs;opencv_imgproc;opencv_ml;opencv_objdetect;opencv_photo;opencv_stitching;opencv_video;opencv_videoio
${OpenCV_INCLUDE_DIRS}
就是 install 目录下的 include 文件夹,${OpenCV_LIBS}
则是 install/x64/mingw/bin 目录下的动态库:
当然一切的前提都是要指定 OpenCV 的路径,如果没有指定或者指定错误,会有相关的提示:
运行代码可能会发生崩溃,此时可以做如下尝试:
- 将前面编译 OpenCV 的编译工具,比如 TDM-GCC 添加到 Toolchains 中,并指定 Build Tool 为编译 OpenCV 时使用的 ming32-make:
- 点击 Edit Configurations:
将编译 OpenCV 并安装后的 install 目录下的 bin 目录添加到 Environment variables 和 Working directory 中:
3.3 Android AS 配置
解压 OpenCV 的 Android SDK,在如图所示的目录中拿到动态库文件 libopencv_java4.so:
至于头文件,Android SDK 中并没有提供,因此我们要使用 Windows 版本提供的通用头文件:
在 AS 模块的 CMakeLists.txt 中引入头文件和库文件:
cmake_minimum_required(VERSION 3.22.1)
project("opencv")
# 导入头文件
include_directories(${CMAKE_SOURCE_DIR}/opencv/include)
# 导入库文件
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_SOURCE_DIR}/libs/${CMAKE_ANDROID_ARCH_ABI}")
add_library(
opencv
SHARED
native-lib.cpp)
target_link_libraries(
opencv
log
opencv_java4 # 链接 OpenCV 动态库
android # 如果要在 Native 层通过 ANativeWindow 渲染,需要链接 libandroid
)
在模块的 build.gradle 中添加 CPU 架构类型的过滤:
android {
...
defaultConfig {
...
externalNativeBuild {
cmake {
abiFilters 'armeabi-v7a'
}
}
ndk {
abiFilters 'armeabi-v7a'
}
}
sourceSets {
main {
jniLibs.srcDirs("src/main/cpp/opencv/libs")
}
}
...
}
这里要特意说一下 sourceSets 的配置。由于 Android 项目的库文件通常位于 jniLibs
目录下,并且 Gradle 默认会将该目录下的库文件复制到 APK 中的正确位置。这种默认行为可以适用于大多数情况。但是现在我们将库文件放在了 /src/main/cpp/opencv/libs/ 目录下:
此时需要显式地配置 Gradle 来处理库文件的复制和加载,就像上面那样在 sourceSets 中指定 jniLibs 的路径,以保证库可以被正确地复制到 APK 中并在运行时加载。否则在执行 System.loadLibrary() 去加载 Native 的动态库时,会抛出异常说找不到 OpenCV 的动态库 libopencv_java4.so: