Opencv图像操作

从文件读取/写入图像,访问像素,原始操作,可视化图像

输入/输出

从文件中加载图像:

Mat img = imread(filename)

把自定义读入的彩色照片变成灰度图像:

Mat img = imread(filename, IMREAD_GRAYSCALE);

写入图像到文件中:

imwrite(filename, img);
文件的格式由其扩展名决定
使用imdecode和imencode从内存中读取和写入,而不是磁盘中

图像的基本操作

访问像素强度

为了获得像素强度值,必须知道图像的类型和通道数。以下是单通道灰度图像(类型8UC1)和像素坐标x和y的示例:

Scalar intensity = img.at<uchar>(y, x);

intensity.val [0]包含0到255之间的值。请注意x和y的顺序。由于OpenCV中的图像由与矩阵相同的结构表示,所以对于这两种情况,我们使用相同的约定 - 基于0的行索引(或y坐标)首先出现,并且基于0的列索引(或x坐标)跟随它。或者,可以使用以下符号:

Scalar intensity = img.at<uchar>(Point(x, y));

现在让我们考虑使用BGR颜色排序的3通道图像(由imread返回的默认格式):

Vec3b intensity = img.at<Vec3b>(y, x);
uchar blue = intensity.val[0];
uchar green = intensity.val[1];
uchar red = intensity.val[2];

您可以使用相同的浮点图像方法(例如,您可以通过在3通道图像上运行Sobel来获取此类图像):

Vec3f intensity = img.at<Vec3f>(y, x);
float blue = intensity.val[0];
float green = intensity.val[1];
float red = intensity.val[2];

可以使用相同的方法来改变像素强度:

img.at<uchar>(y, x) = 128;

OpenCV中有一些功能,特别是来自calib3d模块,如projectPoints,它以Mat的形式获取2D或3D数组。矩阵应该只包含一列,每行对应一个点,矩阵类型应相应为32FC2或32FC3。这样一个矩阵可以很容易地构造成std::vector:

vector<Point2f> points;
//... fill the array
Mat pointsMat = Mat(points);

可以使用相同的方法Mat :: at访问此矩阵中的一个点:

Point2f point = pointsMat.at<Point2f>(i, 0);
内存管理和引用计数

Mat是一种保持矩阵/图像特征(行和列数,数据类型等)和指向数据的指针的结构。所以没有什么可以阻止我们对同一个数据对应几个Mat的实例。Mat保留一个引用计数,用于告知当Mat的特定实例被破坏时是否必须释放数据。以下是创建两个矩阵而不复制数据的示例:

std::vector<Point3f> points;
// .. fill the array
Mat pointsMat = Mat(points).reshape(1);

因此,我们得到一个32FC1矩阵与3列而不是32FC3矩阵与1列。pointsMat使用点数据,销毁时不会释放内存。然而,在这种特殊情况下,开发人员必须确保点的生命周期比pointMat长。如果我们需要复制数据,可以使用例如cv :: Mat :: copyTocv :: Mat :: clone

Mat img = imread("image.jpg");
Mat img1 = img.clone();

相反,使用C API,必须由开发人员创建输出图像,可以向每个功能提供空输出Mat。每个实现都为目标矩阵调用Mat :: create。如果矩阵为空,则此方法分配数据。如果它不是空且具有正确的大小和类型,该方法什么也不做。但是,如果大小或类型与输入参数不同,则数据将被释放(丢失)并分配新的数据。例如:

Mat img = imread("image.jpg");
Mat sobelx;
Sobel(img, sobelx, CV_32F, 1, 0);

在矩阵上定义了一些方便的操作符。例如,我们可以从现有的灰度图像“img”中获取黑色图像:

img = Scalar(0);

选择感兴趣的区域:

Rect r(10, 10, 100, 100);
Mat smallImg = img(r);

从Mat到C API数据结构的转换:

Mat img = imread("image.jpg");
IplImage img1 = img;
CvMat m = img;

注意,这里没有数据复制。

从颜色转换成灰度级:

Mat img = imread("image.jpg"); // loading a 8UC3 image
Mat grey;
cvtColor(img, grey, COLOR_BGR2GRAY);

将图像类型从8UC1更改为32FC1:

src.convertTo(dst, CV_32F);

可视化图像

在开发过程中看到算法的中间结果是非常有用的。OpenCV提供了可视化图像的便捷方式。可以使用以下方式显示8U图像:

Mat img = imread("image.jpg");
namedWindow("image", WINDOW_AUTOSIZE);
imshow("image", img);
waitKey();

waitKey()调用会启动一个消息传递周期,等待“图像”窗口中的关键stroke。32F图像需要转换为8U型

Mat img = imread("image.jpg");
Mat grey;
cvtColor(img, grey, COLOR_BGR2GRAY);
Mat sobelx;
Sobel(grey, sobelx, CV_32F, 1, 0);
double minVal, maxVal;
minMaxLoc(sobelx, &minVal, &maxVal); //find minimum and maximum intensities
Mat draw;