Opencv之saturate_cast防止數據溢出

在OpenCV學習中經常看見saturate_cast的使用,下面的代碼會展示它的作用,這個功能實現的是兩個圖像融合,作者提供了saturate_case的作用,同時比較了兩個方法的運行時間,後面會給出代碼和結果。

也可以原文 http://blog.csdn.net/mjlsuccess/article/details/12401839

一、staturate_cast的原理

大致的原理應該如
if(data<0) 
        data=0; 
elseif(data>255) 
    data=255;

二、關鍵語句,同時保護與不保護可以自己修改

//使用圖像混合例子中的C語言版本演示 
for (int i=0; i<src1.rows; i++) 
{ 
   const uchar* src1_ptr = src1.ptr<uchar>(i); 
   const uchar* src2_ptr = src2.ptr<uchar>(i); 
   uchar* dst_ptr  =dst.ptr<uchar>(i); 
   for (int j=0; j<src1.cols*nChannels; j++) 
   { 
       dst_ptr[j] = saturate_cast<uchar>(src1_ptr[j]*alpha +src2_ptr[j]*beta + gama);//gama = -100, alpha = beta = 0.5 
//     dst_ptr[j] = (src1_ptr[j]*alpha + src2_ptr[j]*beta + gama); 
   } 
} 
imshow("output2",dst);


//沒加入溢出保護 
   for (int i=0; i<src1.rows; i++) 
   { 
       const uchar* src1_ptr = src1.ptr<uchar>(i); 
       const uchar* src2_ptr = src2.ptr<uchar>(i); 
       uchar* dst_ptr  =dst.ptr<uchar>(i); 
       for (int j=0; j<src1.cols*nChannels; j++) 
       { 
//         dst_ptr[j] = saturate_cast<uchar>(src1_ptr[j]*alpha +src2_ptr[j]*beta + gama);//gama = -100, alpha = beta = 0.5 
           dst_ptr[j] = (src1_ptr[j]*alpha + src2_ptr[j]*beta + gama); 
       } 
   } 
imshow("output2",dst);

三、整體代碼

#include "StdAfx.h"  
#include "blending.h"  
using namespace std;  
using namespace cv;  
void blending_test()  
{  
    Mat src1, src2, dst;  
    double alpha = 0.5;  
    double beta = 1-alpha;  
  
    src1 = imread("LinuxLogo.jpg");  
    src2 = imread("WindowsLogo.jpg");  
  
    if(!src1.data) cout<<"error loading src1"<<endl;  
    if(!src2.data) cout<<"Error loading src2"<<endl;  
  
    addWeighted(src1, alpha, src2, beta, 0.0, dst);  
  
    imshow("output1", dst);  
  
//  waitKey(0);  
}  
//C語言自己實現  
void blending()  
{  
    Mat src1, src2, dst;  
    double alpha = 0.5;  
    double beta = 1-alpha;  
    double gama = 0;  
  
    src1 = imread("LinuxLogo.jpg");  
    src2 = imread("WindowsLogo.jpg");  
    //判斷兩幅圖片是否相同  
    CV_Assert(src1.depth() == CV_8U);  
    CV_Assert(src1.depth() == src2.depth());  
    CV_Assert(src1.size() == src2.size());  
    //爲dst申請內存  
    dst.create(src1.size(), src1.type());  
  
    const int nChannels = src1.channels();  
  
    if(!src1.data) cout<<"error loading src1"<<endl;  
    if(!src2.data) cout<<"Error loading src2"<<endl;  
  
    for (int i=0; i<src1.rows; i++)  
    {  
        const uchar* src1_ptr = src1.ptr<uchar>(i);  
        const uchar* src2_ptr = src2.ptr<uchar>(i);  
        uchar* dst_ptr  = dst.ptr<uchar>(i);  
        for (int j=0; j<src1.cols*nChannels; j++)  
        {  
            dst_ptr[j] = src1_ptr[j]*alpha + src2_ptr[j]*beta + gama;  
        }  
    }  
    imshow("output2",dst);  
  
//  cvWaitKey(0);  
}  

int main(int argc, char* argv[])   
{   
    double t;  
    t = (double)getTickCount();   
    blending_test();      
    t = 1000*((double)getTickCount() - t)/getTickFrequency();  
    cout<<"庫函數時間:"<<t<<endl;  
  
    t = (double)getTickCount();  
    blending();  
    t = 1000*((double)getTickCount() - t)/getTickFrequency();  
    cout<<"C語言實現:"<<t<<endl;  
  
    cvWaitKey(0);  
    return 0;  
}  

結果如下:第一副是加保護,第二幅是不加保護,第三幅是運行時間的對比

                                                                                                                                                                                                                                       


                                              

發佈了135 篇原創文章 · 獲贊 274 · 訪問量 130萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章