Edge Detection
Giacomo Boracchi
Contents
- Building filters as smoothing + differentiating filters
- Differentiating Filters
- Compute image Principal Derivative (i.e. directional derivatives along axis)
- Hard Threshold - Play on the choice of the Threshold T
- morphological operations on the mask
- compute edges using Canny method (it includes nonmax suppression and histeresis thresholding)
close all
clear
clc
Building filters as smoothing + differentiating filters
disp('differentiating filters') diffx=[1 -1] diffy = diffx' %smoothing filters Previtt sx=ones(2,3); sy=sx'; % build Previtt derivative filters disp('derivative filters Previtt') dx=conv2(sy , diffx) dy=conv2(sx , diffy) %smoothing filters Sobel sx=[1 2 1 ; 1 2 1]; sy=sx'; % Build Sobel derivative filters disp(' derivative filters Sobel') dx=conv2(sy,diffx) dy=conv2(sx,diffy)
differentiating filters diffx = 1 -1 diffy = 1 -1 derivative filters Previtt dx = 1 0 -1 1 0 -1 1 0 -1 dy = 1 1 1 0 0 0 -1 -1 -1 derivative filters Sobel dx = 1 0 -1 2 0 -2 1 0 -1 dy = 1 2 1 0 0 0 -1 -2 -1
Differentiating Filters
y = im2double(imread('image_lena512.png')); figure(1), imshow(y),title('original image'); % compute gradient components (horizontal and vertical derivatives) Gx=conv2(y , dx , 'same'); Gy=conv2(y , dy , 'same'); figure(3),imshow(Gx, []),title('horizontal derivative') figure(4),imshow(Gy, []),title('vertical derivative') % Gradient Norm M=sqrt(Gx.^2 + Gy.^2); figure(5),imshow(M,[]),title('gradient magnitude')
data:image/s3,"s3://crabby-images/ddc14/ddc14701c5610f6526d8273f5e140d866949e67a" alt=""
data:image/s3,"s3://crabby-images/64a8b/64a8be0dd82f6cb3df118aae32d664554b93e1ce" alt=""
data:image/s3,"s3://crabby-images/6ec5b/6ec5b2da6e575449f713ff0e8a34768ebf5bb351" alt=""
data:image/s3,"s3://crabby-images/82610/82610175a84ba1d99d34ed4f428ee689fb7de420" alt=""
Compute image Principal Derivative (i.e. directional derivatives along axis)
% build the gradient image as a two plane image: the first plane contains horizontal derivatives, the second plane the % vertical ones Grad=zeros(size(y,1),size(y,2),2); Grad(:,:,1)=Gx; Grad(:,:,2)=Gy; % compute Gradient norm Norm_Grad = sqrt(Grad(: , : , 1) .^ 2 + Grad(: , : , 2) .^ 2); BORDER = 3; % remove boundaries as these are affected by zero padding Norm_Grad(1 : BORDER, :) = 0; Norm_Grad(end - BORDER : end, :) = 0; Norm_Grad(:, 1 : BORDER) = 0; Norm_Grad(:, end - BORDER : end) = 0; % we change the sign of the derivative because the y axis is "increasing downwards", since in Matlab it correspodns to the row index Dir_Grad=atand(- sign(Grad(:,:,1)).*Grad(:,:,2) ./(abs(Grad(:,:,1))+eps)); figure(2),imshow(Norm_Grad,[]),title('norm'); figure(3),imshow(Dir_Grad,[]),title('directions');
data:image/s3,"s3://crabby-images/1e04f/1e04f36ccab312708516dd144695b5b32c30bb34" alt=""
data:image/s3,"s3://crabby-images/3b977/3b9778a864bfc136cef1112f622b0ca13195249f" alt=""
Hard Threshold - Play on the choice of the Threshold T
% Binary Threshold - Play on the choice of the Threshold T T = 5 * median(Norm_Grad(:)); % or choose the treshold using Otzu method (which is however meant for grayscale images, that are % assumed to contain two classes of pixels, thus that their intensity follow a bi-modal histogram) % [T , eff] = graythresh(Norm_Grad(:)); % arbitrary set the threshold % T = quantile(Norm_Grad(:), 0.9); % 0.9 percentile % compute the "gradient mask" by thresholding gradient magnitude mask = Norm_Grad > T; figure(4), imshow(mask), title('gradient mask') % compute the histogram of gradient norm intensities [hh , hh_bins] = hist(Norm_Grad(:) , 1000); % show the histogram with the tresholded value figure(7), bar(hh_bins , hh) hold on plot([T , T] , [min(hh) , max(hh)] , 'g' ,'LineWidth' , 3) hold off % perform Hard Tresholding using a mask Norm_Grad_Binary = double(Norm_Grad > T); Norm_Grad_Hard=Norm_Grad .* Norm_Grad_Binary; % show Gradient Norm figure(2),imshow(Norm_Grad,[]),title('Gradient norm'); % show the mask figure(3),imshow(Norm_Grad_Binary,[]),title('Binary Thresholded gradient norm'); % show HT results figure(4),imshow(Norm_Grad_Hard,[]),title(' Hard Thresholded gradient norm'); % show gradient directions in areas where gradient norm is large (i.e. over the gradient mask) figure(5),imshow(Norm_Grad_Binary.*Dir_Grad,[])
data:image/s3,"s3://crabby-images/5a40e/5a40e1dc93a907d8473572f596dcb0a0dc82682e" alt=""
data:image/s3,"s3://crabby-images/f960b/f960b3a3cfb1d974bb5718f5c145457ff09b74bf" alt=""
data:image/s3,"s3://crabby-images/d9c7d/d9c7dcef64396e404113692311387b6341e2d333" alt=""
data:image/s3,"s3://crabby-images/01d57/01d578798f04dda0b24d8c400be04edc2dcb22fd" alt=""
data:image/s3,"s3://crabby-images/ddc13/ddc13963e01ed8c43d35967d73f52d62157fc83d" alt=""
morphological operations on the mask
% dilate via convolution % Norm_Grad_mask=double(conv2(Norm_Grad_Binary,ones(3),'same')>0); Norm_Grad_mask = imerode(Norm_Grad_Binary , ones(3)); Norm_Grad_mask = imdilate(Norm_Grad_mask , ones(5)); figure(3),imshow(Norm_Grad_mask),title('Mask on gradient norm'); % show directions on the dilated mask figure(4),imshow(Dir_Grad.*Norm_Grad_mask,[]),title('Gradient direction'),impixelinfo;
data:image/s3,"s3://crabby-images/c070f/c070f1591713360d98323203b8a8509cc0c7fba3" alt=""
data:image/s3,"s3://crabby-images/f74d3/f74d309393b290cc21e294682efc451dcd1a24b8" alt=""
compute edges using Canny method (it includes nonmax suppression and histeresis thresholding)
edgs = edge(y, 'canny');
figure(6), imshow([y, edgs])
data:image/s3,"s3://crabby-images/14b72/14b726f681e606e7ec0161c0ae4207cd1270ba79" alt=""