图像差分的方法

差分图像的几个方法

2017/7/25


在处理图像的时候,特别是处理视频流图像的时候,往往会用到图像差分的方法。顾名思义,图像差分,就是把两幅图像的对应像素值相减,以削弱图像的相似部分,突出显示图像的变化部分。例如,差分图像往往能够检测出运动目标的轮廓,能够提取出闪烁导管的轨迹等等。

 

         那么,该如何对图像进行差分呢,或者说输入一个视频流,如何进行差分处理呢。

         懂点OpenCV的同学可能就要说了,opencv里不是有现成的函数实现差分的吗,直接拿来用就可以了,废话那么多作甚。没错,OpenCV里跟差分相关的函数有两个,一个是cvSub()函数,一个是cvAbsDiff()函数。先来看看这两个函数的参数。

 

void cvSub(const CvArr* src1, const CvArr* src2, CvArr* dst,const CvArr* mask=NULL);

 

两个输入图像src1src2和一个输出图像dst具有相同的类型和大小。cvSub适用于IplImage以及cvMat两种结构。一个简单的例子如下:

#include <opencv2/opencv.hpp>

 

int main()

{

    IplImage*src1 = cvLoadImage("E:\\testvideo\\test1.png");

    IplImage*src2 = cvLoadImage("E:\\testvideo\\test2.png");

    assert(src1);

    IplImage*dst = cvCreateImage(cvGetSize(src1),src1->depth,src1->nChannels);

    cvSub(src1,src2,dst);

    cvShowImage("1",src1);

    cvShowImage("2",src2);

    cvShowImage("dst",dst);

    cvWaitKey(0);

}

图像差分的方法


图像差分的方法

执行结果:

图像差分的方法

如果将代码cvSub(src1,src2,dst);改为cvSub(src2,src1,dst);则差分结果为:

图像差分的方法

这说明,cvSub()函数是直接将两者的像素值相减,差值小于零的归一到零处理,而没有取差的绝对值。同时,也说明了cvSub()不仅支持灰度图像,也支持三通道图像。

 

cvAbsDiff()函数计算了两幅图像中差的绝对值。其参数跟cvSub()函数类似,如下所示,

 

void cvAbsDiff(const CvArr* src1, const CvArr* src2, CvArr* dst );

 

所以cvAbsDiff(src2,src1,dst);和cvAbsDiff(src1,src2,dst);的执行结果一样,如下:

图像差分的方法


但是,很多人肯定想着自己实现代码,并进行优化,

/*图像的差分

*要求输入输出图像有相同的格式和大小

*/

void  cvSub(IplImage* src1,IplImage*src2,IplImage* dst)

{

    IplImage*src1_gray = cvCreateImage(cvGetSize(src1),8,1);

    IplImage*src2_gray = cvCreateImage(cvGetSize(src2),8,1);

    cvCvtColor(src1,src1_gray,CV_RGB2GRAY);

    cvCvtColor(src2,src2_gray,CV_RGB2GRAY);

    CvScalarpixel;

    for (int i = 0;i <src1->height; i++)

        for (int j = 0; j< src1->width; j++)

        {

            CvScalarp1 = cvGet2D(src1_gray,i,j);

            CvScalarp2 = cvGet2D(src2_gray,i,j);

            pixel.val[0]= abs(p1.val[0] - p2.val[0])*120/(p1.val[0]); //相对灰度值

            cvSet2D(dst,i,j,pixel);

        }

        cvReleaseImage(&src1_gray);

        cvReleaseImage(&src2_gray);

        cvShowImage("result",dst);

}

int main()

{

    IplImage*src1 = cvLoadImage("D:\\test1.png");

    IplImage*src2 = cvLoadImage("D:\\test2.png");

    assert(src1);

    IplImage*dst = cvCreateImage(cvGetSize(src1), src1->depth, src1->nChannels);

    cvSub(src1, src2, dst);

    cvShowImage("1", src1);

    cvShowImage("2", src2);

    cvShowImage("dst", dst);

    cvWaitKey(0);

}

运行效果如下:图像差分的方法


//图像差分
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
#include "opencv2/opencv.hpp"
#include "opencv2/imgproc.hpp"
#include "opencv2/highgui.hpp"
#include "opencv2/ml.hpp"

using namespace std;

void Image_Minus(IplImage *X, IplImage *Y, IplImage *X_Y)
{
    //图像差分函数,将图像1中像素和图像2中对应像素想减,要求X、Y、X_Y大小相同
    int i, j, width, height, step, chanel;
    unsigned char *dataX, *dataY, *dataX_Y;

    width = X->width;
    height = X->height;

    //存入矩阵数据
    dataX = (unsigned char *)X->imageData;
    dataY = (unsigned char *)Y->imageData;
    dataX_Y = (unsigned char *)X_Y->imageData;

    //计算步长
    step = X->widthStep / sizeof(char);
    chanel = X->nChannels;

    //一个个数据处理
    for (i = 0; i<height; i++)
    for (j = 0; j<width*chanel; j++)
        dataX_Y[i*step + j] = abs(dataX[i*step + j] - dataY[i*step + j]);
}


int main()
{
    IplImage* pImgX;
    IplImage* pImgY;
    IplImage* pImgX_Y;
    CvSize dest_size;
    pImgX = cvLoadImage("D:\\test1.png", -1);
    pImgY = cvLoadImage("D:\\test2.png", -1);

    if (pImgX == 0 || pImgY == 0)
    {
        printf("载入文件失败!/n");
        return -1;
    }
    dest_size.width = pImgX->width;
    dest_size.height = pImgX->height;

    cout << "width == " << dest_size.width << endl;
    cout << "height == " << dest_size.height << endl;

    pImgX_Y = cvCreateImage(dest_size, pImgX->depth, pImgX->nChannels);

    //图像差分,最最关键的一步
    Image_Minus(pImgX, pImgY, pImgX_Y);


    //创建窗口
    cvNamedWindow("Picture X:", 1);
    cvNamedWindow("Picture Y:", 1);
    cvNamedWindow("Picture X-Y:", 1);
    //显示图像
    cvShowImage("Picture X:", pImgX);
    cvShowImage("Picture Y:", pImgY);
    cvShowImage("Picture X-Y:", pImgX_Y);
    cvWaitKey(0);
    //销毁窗口
    cvDestroyWindow("Picture X:");
    cvDestroyWindow("Picture Y:");
    cvDestroyWindow("Picture X-Y:");
    //释放图像
    cvReleaseImage(&pImgX);
    cvReleaseImage(&pImgY);
    cvReleaseImage(&pImgX_Y);
    return 0;
}

效果如下图:

图像差分的方法