美文网首页OpenCv
Opencv图像操作

Opencv图像操作

作者: 寻找时光机 | 来源:发表于2017-09-26 22:54 被阅读84次

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

    输入/输出

    从文件中加载图像:

    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;
    sobelx.convertTo(draw, CV_8U, 255.0 / (maxVal - minVal), -minVal * 255.0 / (maxVal - minVal));
    namedWindow("image", WINDOW_AUTOSIZE);
    imshow("image", draw);
    waitKey();
    

    相关文章

      网友评论

        本文标题:Opencv图像操作

        本文链接:https://www.haomeiwen.com/subject/gkgeextx.html