上一篇文章,我們講解了圖像處理中的膨脹和腐蝕函數(shù),這篇文章將做邊緣梯度計算函數(shù)。
圖像的邊緣
圖像的邊緣從數(shù)學上是如何表示的呢?
圖像的邊緣上,鄰近的像素值應(yīng)當顯著地改變了。而在數(shù)學上,導數(shù)是表示改變快慢的一種方法。梯度值的大變預示著圖像中內(nèi)容的顯著變化了。
用更加形象的圖像來解釋,假設(shè)我們有一張一維圖形。下圖中灰度值的“躍升”表示邊緣的存在:
使用一階微分求導我們可以更加清晰的看到邊緣“躍升”的存在(這里顯示為高峰值):
由此我們可以得出:邊緣可以通過定位梯度值大于鄰域的相素的方法找到。
近似梯度
比如內(nèi)核為3時。
首先對x方向計算近似導數(shù):
然后對y方向計算近似導數(shù):
然后計算梯度:
當然你也可以寫成:
函數(shù)實現(xiàn)}
GRAY216IC1Filter(__src, size, height, width, kernel, dstData, __borderType);
}else{
error(arguments.callee, UNSPPORT_DATA_TYPE/* {line} */);
}
return dst;
};
這里只提供了內(nèi)核大小為3和5的Sobel算子,主要原因是7或以上的內(nèi)核計算就比較慢了。
輸出一個單通道的16位有符號整數(shù)矩陣。
代碼如下:
function GRAY216IC1Filter(__src, size, height, width, kernel, dstData, __borderType){
var start = size >> 1;
var withBorderMat = copyMakeBorder(__src, start, start, 0, 0, __borderType);
var mData = withBorderMat.data,
mWidth = withBorderMat.col;
var i, j, y, x, c;
var newValue, nowX, offsetY, offsetI;
for(i = height; i--;){
offsetI = i * width;
for(j = width; j--;){
newValue = 0;
for(y = size; y--;){
offsetY = (y + i) * mWidth;
for(x = size; x--;){
nowX = x + j;
newValue += (mData[offsetY + nowX] * kernel[y * size + x]);
}
}
dstData[j + offsetI] = newValue;
}
}
}
然后把內(nèi)核和矩陣交給這個濾波器處理,就OK了。
把這個濾波器獨立出來的原因是,可以給其他類似的計算邊緣函數(shù)使用,比如Laplacian和Scharr算子。
轉(zhuǎn)為無符號8位整數(shù)由于Sobel算子算出來的是16位有符號整數(shù),無法顯示成圖片,所以我們需要一個函數(shù)來將其轉(zhuǎn)為無符號8位整數(shù)矩陣。
convertScaleAbs函數(shù)是將每個元素取絕對值,然后放到Int8Array數(shù)組里面,由于在賦值時候大于255的數(shù)會自動轉(zhuǎn)成255,而小于0的數(shù)會自動轉(zhuǎn)成0,所以不需要我們做一個函數(shù)來負責這一工作。
代碼如下:
function convertScaleAbs(__src, __dst){
__src || error(arguments.callee, IS_UNDEFINED_OR_NULL/* {line} */);
var height = __src.row,
width = __src.col,
channel = __src.channel,
sData = __src.data;
if(!__dst){
if(channel === 1)
dst = new Mat(height, width, CV_GRAY);
else if(channel === 4)
dst = new Mat(height, width, CV_RGBA);
else
dst = new Mat(height, width, CV_8I, channel);
}else{
dst = __dst;
}
var dData = dst.data;
var i, j, c;
for(i = height; i--;){
for(j = width * channel; j--;){
dData[i * width * channel + j] = Math.abs(sData[i * width * channel + j]);
}
}
return dst;
}
我們還需要一個函數(shù)將x方向梯度計算值和y方向梯度計算值疊加起來。
代碼如下:
var addWeighted = function(__src1, __alpha, __src2, __beta, __gamma, __dst){
(__src1 && __src2) || error(arguments.callee, IS_UNDEFINED_OR_NULL/* {line} */);
var height = __src1.row,
width = __src1.col,
alpha = __alpha || 0,
beta = __beta || 0,
channel = __src1.channel,
gamma = __gamma || 0;
if(height !== __src2.row || width !== __src2.col || channel !== __src2.channel){
error(arguments.callee, "Src2 must be the same size and channel number as src1!"/* {line} */);
return null;
}
if(!__dst){
if(__src1.type.match(/CV\_\d+/))
dst = new Mat(height, width, __src1.depth(), channel);
else
dst = new Mat(height, width, __src1.depth());
}else{
dst = __dst;
}
var dData = dst.data,
s1Data = __src1.data,
s2Data = __src2.data;
var i;
for(i = height * width * channel; i--;)
dData[i] = __alpha * s1Data[i] + __beta * s2Data[i] + gamma;
return dst;
};
這個函數(shù)很簡單,實際上只是對兩個矩陣的對應(yīng)元素按固定比例相加而已。 效果圖
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com