数字图像处理
数字图像处理—图像分割
概述:
1、在对图像的研究和应用中,人们往往仅对图像中的某些部分感兴趣,这些部分一般称为目标或前景。
2、为了辨识和分析目标,需要将有关区域分离提取出来,在此基础上对目标进一步利用,如进行特征提取和测量。
3、图像分割就是指把图像分成各具特性的区域并提取出感兴趣目标的技术和过程。
4、特性可以是灰度、颜色、纹理等,目标可以对应单个区域,也可以对应多个区域。
5、图像分割是图像识别和图像理解的基本前提步骤。
(一)点、线和边缘检测
1.1 点检测
使用如下图所示的模板,如果:
在这里插入图片描述
|R|>=T
则在模板中心位置检测到一个点。其中,T是阈值,R是模板计算值。
基本思想: 如果一个孤立点与它周围的点不 同,则可以使用上述模板进行检测。
注意: 如果模板响应为0,则表示在灰度级为常数的区域。
函数:
g = abs(imfilter(tofloat(f), w)) >= T;
其中,f 是输入图像,w 是适合点检测的模板,g 是包含检测点的图像。代码如下:
f = imread("C:\Users\madster\Downloads\DIP3E_CH05_Original_Images\Fig0524(b)(blurred-impulse).tif");
w = [-1 -1 -1;-1 8 -1;-1 -1 -1]; % 点检测掩模
g = abs(imfilter(double(f),w));
T = max(g(:));
g = g>=T;
subplot(1,2,1);imshow(f);title('(a)原图像');
subplot(1,2,2);imshow(g);title('(b)点检测');
分析:
图 (a)显示了一幅图像f,在球体的东北象限有一个几乎不可见的黑点,检测该点,在滤波后的图像g中选择最大值作为 T值。然后在g中寻找所有的g>=T的点。假设所有的点是孤立的镶嵌在恒定或是近似恒定的背景上。我们可识别能给出最大响应的点。因为在选择g中的最大值作为T值的情况下,在g中不存在比T值大的点。我们使用>=算子(代替=)定 义一致性。如图 (b)所示,其中有一个孤立点,该点使用 T值置为max(g (: ))且满足g>=T 的条件。
1.2 线检测
通过比较典型模板的计算值,确定一个点是否在某个方向的线上。
用R1,R2,R3和R4分别代表水平、45°、垂直和 -45°方向线的模板响应,在图像中心的点,如果
|Ri|>|Rj|,j不等于i,则此点被认为与在模板i方向上的线更相关。
例:如果|R i |>|R j |,j=2,3,4,
则该点与水平线有更大的关联。
在灰度恒定的区域,上述4个模板的响应为零。
检测指定方向的线 :
代码如下:
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH09\Fig0905(a)(wirebond-mask).tif"); % 图像大小:486×486
w = [2 -1 -1;-1 2 -1;-1 -1 2]; % +45°方向检测线
g = imfilter(double(f),w);
gtop = g(1:120,1:120); % 左上角区域
gtop = pixeldup(gtop,4); % 通过复制像素将图像扩大gtop*4倍
gbot = g(end-119:end,end-119:end); % 右下角区域
gbot = pixeldup(gbot,4);
g1 = abs(g); % 检测图的绝对值
T = max(g1(:));
g2 = g1>=T;
subplot(3,2,1);imshow(f);title('(a)连线模板图像');
subplot(3,2,2);imshow(g,[]);title('(b)+45°线处理后的结果');
subplot(3,2,3);imshow(gtop,[]);title('(c)(b)中左上角的放大效果');
subplot(3,2,4);imshow(gbot,[]);title('(d)(b)中右下角的放大效果');
subplot(3,2,5);imshow(g1,[]);title('(e)(b)的绝对值');
subplot(3,2,6);imshow(g2);title('(f)满足g>=T的所有点');
分析:
图 (a)显示了一幅数字化(二值化)的电子线路连线模板的一部分。图像大小是486×486 像素。
在图 (b)中,比灰色背景暗一点的阴影与负值 相对应。在+45°方向上有两个主要线段,一个在左 上方,另一个在右下方(图©和图(d)显示了这 两个区域的放大的片段)。
注意,在图 (d)中,直线部分比图 ©中的线段要亮得多。原因是:图 (a)中右下方的部分只有一个像素宽,但左上方的部分不是。模板的响应对一个像素宽的部分比较强烈。
图(e)显示了图(b)的绝对值。因为我们对最强的响应感兴趣,所以令 T等于图像中的最大值。图 (f)以白色显示了满足条件g>=T的点,其中的 g是图(e)中的图像。
1.3 使用函数edge的边缘检测
边缘:
边缘是指图像中像元灰度有阶跃变化或屋顶状变化的那些像元的集合。
存在于目标与背景、目标与目标、区域与区域、基元与基元之间。
包含方向、阶跃性质、形状等信息。
是图像识别中抽取的重要属性,对图像识别和分析十分有用。
边缘的分类:
1、阶跃状:位于两边的像素灰度值有明显不同。
2、屋顶状:位于灰度值从增加到减少的转折处。
图解: 边缘的一阶导数在图像由暗变亮的突变位置有一个正的峰值,而在图像由亮变暗的位置有一负的峰值,而在其他位置都为0。
分析: 可用一阶导数的幅度值来检测边缘的存在,幅度峰值对应的一般就是边缘的位置,峰值的正或负就表示边缘处由暗变亮还是由亮变暗。同理,可用二阶导数的过0点检测图像中边缘的存在。二阶导数判断边缘像素在边缘亮的一边,还是暗的一边。
分析:
高斯型函数的目的是对图像进行平滑处理。
拉普拉斯算子的目的是提供一幅用零 交叉确定边缘位置的图像。
平滑处理减少了噪声的影响。
Sobel 边缘检测算子的使用
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH04\Fig0438(a)(bld_600by600).tif"); % 图像大小:486×486
subplot(3,2,1),imshow(f),title('(a)原始图像');
[gv, t] = edge(f,'sobel','vertical');
subplot(3,2,2),imshow(gv),title('(b)Sobel模板处理后结果');
gv = edge(f, 'sobel', 0.15, 'vertical');
subplot(3,2,3),imshow(gv),title('(c)使用指定阈值的结果');
gboth = edge(f, 'sobel', 0.15);
subplot(3,2,4),imshow(gboth),title('(d)指点阈值确定垂直边缘和水平边缘的结果');
w45 = [-2 -1 0;-1 0 1; 0 1 2];
g45 = imfilter(double(f), w45, 'replicate');
T = 0.3 * max(abs(g45(:)));
g45 = g45 >= T;
subplot(3,2,5),imshow(g45),title('(e)-45°方向边缘');
f45= [0 1 2;-1 0 1;-2 -1 0];
h45= imfilter(double(f), f45,'replicate');
T = 0.3 * max(abs(h45(:)));
h45 = h45 >= T;
subplot(3,2,6),imshow(h45),title('(f)+45°方向边缘');
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH04\Fig0438(a)(bld_600by600).tif"); % 图像大小:486×486
[g_sobel_default,ts] = edge(f,'sobel');
[g_log_default,tlog] = edge(f,'log');
[g_canny_default,tc] = edge(f,'canny');
g_sobel_best = edge(f,'sobel',0.05);
g_log_best = edge(f,'log',0.003,2.25);
g_canny_best = edge(f,'canny',[0.04 0.10],1.5);
subplot(3,2,1);imshow(g_sobel_default);title('(a)默认sobel');
subplot(3,2,2);imshow(g_sobel_best);title('(b)最佳sobel');
subplot(3,2,3);imshow(g_log_default);title('(c)默认LoG');
subplot(3,2,4);imshow(g_log_best);title('(d)最佳LoG');
subplot(3,2,5);imshow(g_canny_default);title('(e)默认canny');
subplot(3,2,6);imshow(g_canny_best);title('(f)最佳canny');
感兴趣的主要边缘是建筑的角落、窗户、由亮的砖 结构形成的门口和门本身、屋顶线以及围绕建筑且距离地面三分之二的混凝土带。
对于’log’和’canny’选项,sigma的默认值分别是 2.0 和 1.0。除了 Sobel 图像之 外,通过默认值计算得出的图像与想要得到的清晰边缘图相差较远。
由默认值开始,每个选项的参数分别随早些时候提到的显示主要特性的目标而变化,并尽可能减少不必要的细节。
如图(b)所示,Sobel 得出的结果与我们想要的水泥带和门口的左边缘相差还较远。在 (d)中,LoG的结果相比 Sobel 的结果要好一些,比 LoG的默认值得出的结果要好得多。但是,门口的左边缘没有检测出来,建筑物周围水泥带的两个边也还是不够清晰。Canny得出的结果(图(f))到目前为止要远远好于前两种。
特别注意:门口的左边缘已被清晰检测到,还有混凝土带和其他细节,比如门口上方屋顶的通风栅格。除了检测到要求的特征之外,Canny检测算子还产生了最清晰的边缘图。
(二)使用霍夫变换的线检测
霍夫变换可以用于将边缘像素连接起来得到边界曲线,它的主要优点在于受噪声和曲线间断的影响较小。
图像处理工具箱提供了 3 个与霍夫变换有关的函数。
函数 hough 实现霍夫变换的概念。
函数 houghpeaks 寻找霍夫变换中的峰值(高计数累加器单元)。
函数 houghlines 基于前两个函数的结果,提取原始图像中的线段。
2.1 函数hough
语法:
[H, theta, rho] = hough(f)
[H, theta, rho] = hough(f, ‘ThetaRes’, val1, ‘RhoRes’, val2)
H 是霍夫变换矩阵,theta(以度计)和 rho 是 ρ 和 θ 值向量,在这些值上产生霍夫变换。val1是 0 到 90 的标量,指定了沿 θ 轴霍夫变换的间距(默认是 1),val2 是 0
f = zeros(101, 101);
f(1, 1) = 1; f(101, 1) = 1; f(1, 101) = 1;
f(101, 101) = 1; f(51, 51) = 1;
H = hough(f);imshow(H,[]);
[H, theta, rho] = hough(f);
figure(2)
imshow(H, [], 'XData', theta, 'YData', rho ,'InitialMagnification', 'fit')
axis on, axis normal
xlabel('\theta'), ylabel('\rho')
peaks = houghpeaks(H, NumPeaks)
peaks = houghpeaks(…, ‘Threshold’, val1, ‘NHoodSize’, val2)
“…”指出来自默认语法和 peaks 的输入是持有峰值行和列坐标的 Q×2 大小的矩 阵。Q的范围是 0 到NumPeaks,H是霍夫变换矩阵。参数val1是非负的标量,指定了 H中 的什么值被考虑为峰值;val1可以从 0 到Inf变化,默认值是0.5*max(H( : ))。参数val2 是奇整数的两元素矢量,指定量围绕峰值的邻域大小。
2.3 函数houghlines
语法:
lines = houghlines(f, theta, rho, peaks)
lines = houghlines(…, ‘FillGap’, val1, ‘MinLength’, val2)
theta和rho是来自函数hough的输出,peaks是函数houghpeaks的输出。输出lines是结构数组,长度等于找到的线段数。结构中的每个元素可以看成一条线,并含有下列字段:
1、point1:两元素向量[r1, c1],指定了线段终点的行列坐标。
2、point2:两元素向量[r2, c2],指定了线段其他终点的行列坐标。
3、theta:与线相关的霍夫变换的以度计量的角度。
4、rho:与线相关的霍夫变换的ρ轴位置。
5、val1:是正的标量,指定了与相同的霍夫变换相关的两条线段的距离。当两条线段之间的距离小于指定的值时,函数houghlines把线段合并为一条线段(默认的距离是20个像素)。
6、val2:是正的标量,指定合并的线是保留还是丢弃。如果合并的线比val2指定的值短,就丢弃(默认值是40)。
用霍夫变换检测和连接线
在这个例子中,我们用函数hough、houghpeaks和houghlines寻找二值图像f的一组线段。首先,我们用比默认值更好的角间距(用0.2代替1.0)计算和显示霍夫变换:
f=imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1025(a)(building_original).tif");
figure(1)
imshow (f);title('The Original image')
level=graythresh(f);
f=im2bw(f);
[H, theta, rho] = hough(f, 'ThetaResolution', 0.2);
figure(2)
imshow(H, [], 'XData', theta, 'YData', rho, 'InitialMagnification', 'fit')
axis on, axis normal
hold on
xlabel('\theta'), ylabel('\rho')
peaks = houghpeaks(H,2);%检测2个峰值点
plot(theta(peaks(:, 2)), rho(peaks(:, 1)), ...
'linestyle', 'none', 'marker', 's', 'color', 'w')
title('The peak point location')
lines = houghlines(f, theta, rho, peaks);
figure(3)
imshow(f), hold on
for k = 1:length(lines)
xy=[lines(k).point1 ; lines(k).point2];
plot(xy(:,1), xy(:,2), 'LineWidth', 4, 'Color', [.8 .8 .8]);
end
title('Hough-transformation result')
(三)阈值处理
3.1 基本全局阈值处理
基于图像数据自动地选择阈值算法:
(1) 针对全局阈值选择初始估计值T。
(2) 用T 分割图像。这会产生两组像素:G1 由所有灰度值大于T 的像素组成,G2 由所有灰度值小于等于T 的像素组成。
(3) 分别计算G1、G2 区域内的平均灰度值m1 和m2。
(4) 计算出新的阈值:
(5) 重复步骤(2)~(4),直到在连续的重复中,T 的差异比预先设定的参数△T 小为止。
(6) 使用函数im2bw 分割图像:
g = im2bw(f, T/den)
其中,den 是整数(例如一幅8 比特图像的255),是T/den 比率为1 的数值范围内的最大值,正如函数im2bw 要求的那样。
计算全局阈值:
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1038(a)(noisy_fingerprint).tif");
count=0;
T=mean2(f);
done=false;
while ~done
count=count+1;
g=f>T;
Tnext=0.5*(mean(f(g))+mean(f(~g)));
done=abs(T-Tnext)<0.5;
T=Tnext;
end
disp('count的值为:');
disp(['count=',num2str(count)]) %打印输出count的值
disp('T的值为:');
disp(['T=',num2str(T)]) %打印输出T的值
g=im2bw(f,T/255);
figure;subplot(1,3,1);imshow(f);title('(a)带噪声的指纹');
subplot(1,3,2);imhist(f);title('(b)直方图');
subplot(1,3,3);imshow(g);title('(c)用全局阈值分割的结果');
3.2 使用 Otsu’s 方法的最佳全局阈值处理
Otsu算法:取一个最优阈值把原图像分为前景色(A部分)与背景色(B部分),两部分的类间方差越大,说明两部分差别越大,便能有效的分割图像。所以该算法最关键的是找到最优阈值。
语法:
[T, SM] = graythresh(f)
f是输入图像,T是产生的阈值并被规一化到[0,1]中,SM是可分性度量。
f2 = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1039(a)(polymersomes).tif");
subplot(221),imshow(f2),title('原始图像');
subplot(222),imhist(f2),title('原始图像直方图');
count = 0;
T = mean2(f2);
done = false;
while ~done
count = count+1;
g = f2>T;
Tnext = 0.5*(mean(f2(g))+mean(f2(~g)));
done = abs(T-Tnext)<0.5;
T =Tnext;
end
g = im2bw(f2,T/255);
subplot(223),imshow(g,[]),title('用基本全局算法分割结果')
[T,SM] = graythresh(f2);
SM
T*255
g = im2bw(f2,T);
subplot(224),imshow(g),title('用Otsu’s算法分割结果')
3.3 使用图像平滑改进全局阈值处理
噪声会把简单的阈值处理问题变为不能解决的问题。当不能再源头降低噪声且阈值处理是所选择的分割方法时,增强性能的一种常用技术是在阈值处理前先对图像进行平滑。
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1036(b)(gaussian_noise_mean_0_std_10_added).tif");
fn = imnoise(f,'gaussian',0,0.038);
subplot(2,3,1),imshow(fn);title('(a) 带噪图像');
subplot(2,3,2),imhist(fn);title('(b) 直方图');
Tn = graythresh(fn);
gn = im2bw(fn,Tn);
subplot(2,3,3),imshow(gn);title('(c)用 Otsu s 方法得到的结果');
w = fspecial('average',5);
fa = imfilter(fn,w,'replicate');
subplot(2,3,4),imshow(fa);title('(d)用 5×5 均值模板平滑后的图像');
subplot(2,3,5),imhist(fa);title('(e) 直方图');
Ta = graythresh(fa);
ga = im2bw(fa,Ta);
subplot(2,3,6),imshow(ga);title('(f)使用 Otsu s 方法的阈值处理后的结果');
3.4使用边缘改进全局阈值处理
边缘改进的阈值处理:主要是处理那些位于或接近物体和背景间边缘的像素,使得这些像素分离开的操作。
具体算法过程如下:
用一种边缘查找方式计算图像的模板的值。
通过百分比指定阈值。由于计算的边缘模板值中有很多噪声,所以可以将计算值排序,并选择百分比相对高的值(大于百分下的值的阈值)作为阈值。
根据指定的阈值,对第一步的图像边缘的值进行选择。使高于阈值的像素点值为1,低于的值为零,由此选择出部分边缘点的二值图像(模板)。
用刚才计算出来的模板与原图像相乘,获得一幅新的图像。
对新的图像使用otsu进行分割。
用自定义的函数percentile2i来计算灰度值I的对应指定的百分比:
I = percentile2i(h,p)
编写实验一代码:基于梯度边缘信息改进全局阈值处理
f = tofloat(imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1042(a)(septagon_small_noisy_mean_0_stdv_10).tif"));
subplot(231),imshow(f),title('原图像');
subplot(232),imhist(f),title('原图像直方图');
sx = fspecial('sobel');
sy = sx';
gx = imfilter(f,sx,'replicate');
gy = imfilter(f,sy,'replicate');
grad = sqrt(gx.*gx+gy.*gy);
grad = grad/max(grad(:));
h =imhist(grad);
Q = percentile2i(h,0.999);
markerImage = grad>Q;
subplot(233),imshow(markerImage),title('以99.9%进行阈值处理后的梯度幅值图像');
fp = f.*markerImage;
subplot(234),imshow(fp),title('原图像与梯度幅值乘积的图像');
hp = imhist(fp);
hp(1) = 0;
subplot(235),bar(hp),title('原图像与梯度幅值乘积的直方图');
T = otsuthresh(hp);
T*(numel(hp)-1)
g = im2bw(f,T);
subplot(236),imshow(g),title('改进边缘后的图像')
3.4使用边缘改进全局阈值处理
当物体比背景小得多时,他们对直方图的贡献可以或略不计,使用边缘信息可以改进这种情况。
算法:
1.计算一副边缘图像(梯度和拉普拉斯都可以)。
2.指定一个阈值T。
3.用步骤2中的阈值对步骤1中产生的图像进行阈值处理,产生一副二值图像gT(x,y)。在从f(x,y)中选取对应于强边缘像素的下一步中,该图像作为一副模板图像
4.仅用f(x,y)中对应于gT(x,y)中像素为1的位置的像素计算直方图。
5.用步骤4中的直方图全局地分割f(x,y)。
通常指定对应于一个百分位的 T 值,它通常设置一个高值(如第90个百分位),以便边缘图像中只有很少的像素用于阈值的计算。
自定义函数 percentile2i 可实现这一目的。该函数计算一个灰度值 I,它对应于一个指定的百分位 P。语法为:
I = percentile2i(h, P)
1、h 是图像的直方图
2、P 是区间 [0, 1] 内的百分位值
3、I 是对应于第 P 个百分位的灰度级
subplot(231),imshow(f),title('原始图像');
subplot(232),imhist(f),title('原始图像的直方图');
hf = imhist(f);
[Tf SMf] = graythresh(f);
gf = im2bw(f,Tf);
subplot(233),imshow(gf),title('对原始图像进行分割的图像');
w= [-1 -1 -1;-1 8 -1;-1 -1 -1];
lap = abs(imfilter(f,w,'replicate'));
lap = lap/max(lap(:));
h = imhist(lap);
Q = percentile2i(h,0.995);
markerImage = lap >Q;
fp = f.*markerImage;
subplot(234),imshow(fp),title('标记图像与原图像的乘积');
hp = imhist(fp);
hp(1) =0;
subplot(235),bar(hp)
T = otsuthresh(hp);
g = im2bw(f,T);
subplot(236),imshow(g),title('修改边缘后的阈值处理图像')
hon
3.5 基于局部统计的可变阈值处理
在不规则光照情形下,或者在有多个主要物体灰度的情况下(此时全局阈值处理有困难),进行补偿的一种方法是采用可变阈值处理。
计算局部标准差,使用函数 stdfilt:
g = stdfilt(f, nhood)
nhood 是由 0 和 1 组成的数组,其中非零元素指定用于计算局部标准差所用的领域。nhood 的尺寸在每个维度上必须是奇数,默认值是 ones(3)。
对全局和局部阈值处理的比较 :
f = tofloat(imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1043(a)(yeast_USC).tif"));
subplot(2,2,1),imshow(f);title('(a) 酵母细胞的图像');
[TGlobal] = graythresh(f);
gGlobal = im2bw(f, TGlobal);
subplot(2,2,2),imshow(gGlobal);title('(b)用 Otsus 方法分割的图像');
g = localthresh(f, ones(3), 30, 1.5, 'global');
SIG = stdfilt(f, ones(3));
subplot(2,2,3), imshow(SIG, [ ]) ;title('(c) 局部标准差图像');
subplot(2,2,4),imshow(g);title('(d) 用局部阈值处理分割的图像 ');
从背景中分割出细胞来,并且从细胞的主体分出细胞核(内部的亮区域)。这幅图像中有三个主要的灰度级,因此有理由期待这样的分割是可能的。
因为细胞核比细胞本身明显较亮,所以预期围绕细胞核边界的标准差相对较大,而围绕细胞边界的标准差稍微小一些。如图©所示,的确是这种情况。由此得出以下结论:在基于 局部标准差的函数localthresh中,这应该是很有帮助的。
如图 (d)所示,运用了属性的分割是相当有效的。个别细胞已经从背景中分割出来了, 并且细胞核也被完全分割出来了。
当背景接近于常数,并且所有物体的灰度高于或低于背景灰度时,选择全局均值一般会得到较好的结果。
3.7 使用移动平均的图像阈值处理
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1049(a)(spot_shaded_text_image).tif");
subplot(131),imshow(f),title('原始图像');
T =graythresh(f);
g1 = im2bw(f,T);
subplot(132),imshow(g1),title('用otsu全局阈值分割后的图像');
g2 = movingthresh(f,20,0.5);
subplot(133),imshow(g2),title('用移动平均局部阈值分割后的图像');
figure(2)
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1050(a)(sine_shaded_text_image).tif");
subplot(131),imshow(f),title('原始图像');
T =graythresh(f);
g1 = im2bw(f,T);
subplot(132),imshow(g1),title('用otsu全局阈值分割后的图像');
g2 = movingthresh(f,20,0.5);
subplot(133),imshow(g2),title('用移动平均局部阈值分割后的图像');
(四)基于区域的分割
4.1 区域生长
区域生长:根据预先定义的生长准则将像素或子区域组合为更大的区域的过程。
基本方法是从一组 “种子” 点开始,将与种子性质相似的那些邻域像素附加到每个种子上来形成这些生长区域。
语法:
[g, NR, SI, TI] = regiongrow(f, S, T)
S: 可以是一个数组(与 f 大小相同)或一个标量。若 S 是一个数组,则它在种子点的所有坐标处必须包含 1,而在其他地方包含 0。如果 S 是一个标量,则它定义一个灰度值,那么 f 中具有该值的所有点都会成为种子点。
T: 可以是一个数组或一个标量。若 T 是一个数组,则它对 f 中的每个位置包含一个阈值。若 T 是一个标量,则它定义一个全局阈值。S 和 T 所有值必须标定到区间 [0, 1],且与输入图像的类无关。
g: 是分割后的图像。
NR: 是所找到的区域的数量。
SI: 是包含种子点的一幅图像。
TI: 是包含处理连接性前就已通过阈值测试的像素的一幅图像。
使用区域生长检测焊接空隙
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH06\Fig0621(a)(weld-original).tif");
subplot(2,2,1),imshow(f);
title('(a)显示有焊接缺陷的图像');
%函数regiongrow返回的NR为是不同区域的数目,参数SI是一副含有种子点的图像
%TI是包含在经过连通前通过阈值测试的像素
[g,NR,SI,TI]=regiongrow(f,1,0.26);%种子的像素值为255,65为阈值
subplot(2,2,2),imshow(SI);
title('(b)种子点');
subplot(2,2,3),imshow(TI);
title('(c)通过了阈值测试的像素的二值图像(白色)');
subplot(2,2,4),imshow(g);
title('(d)对种子点进行8连通分析后的结果');
图(a)显示了一幅包含几个裂缝(水平的暗区域)和空隙(穿过图像中部的、亮的、白色的 水平方向条纹)的 X射线焊接图像。望使用regiongrow函数来分割相应的焊接缺陷区 域。这些被分割的区域可以用于自动检测这一任务,可以用于包含历史研究的数据库,也可以用于控制自动焊接系统。
图(b)显示了种子点(图像 SI)。在这种情况下,种子点很多,因为种子被指定为在图 像中具有数值 225 的所有点(标定后是1)。
图 ( c)是图像 TI,显示了所有通过阈值测试的 点;也就是说,具有灰度 Zi 且满足︱Zi –S︱≤T 的点。
图 (d)显示了提取图 ( c)中所有连接到种子点的像素的结果。这是分割后的图像 g。通过将这幅图像与原始图像进行比较,区域生长过程确实以合理的精确度分割了焊接的缺陷这一点是很明显的。
4.2 区域分离和聚合
将一幅图像细分为一组任意的不相交区域,然后聚合或分离这些区域。过程如下:
1、令 R 表示整幅图像区域,选择一个属性 P。将满足 P(R i )=False 的任何区域 R i 分离为 4 个不相交的象限区域。
2、当不可能进一步分离时,聚合满足条件 P(R i u R j) = True 的任意两个相邻区域 R i ,R j 。
3、当无法进一步聚合时,停止操作。
工具箱中实现四叉树分解的函数是 qtdecomp。
Z = qtdecomp(f, @split_test, parameters)
1、Z 是包括四叉树结构的一个稀疏矩阵。若 Z(k, m) 是非零的,则 Z(k, m) 是分解的一个块的左上角,且该块的大小是 Z(k, m) 。
2、split_test 用于确定一个区域是否进行分离。
函数 qtgetblk 可得到四叉树分解中的实际四分区域像素值。
[vals, r, c] = qtgetblk(f, Z, m)
1、vals 是一个数组,包含 f 的四叉树分解中的尺寸为 m*m 的块的值。
2、Z 是由 qtdecomp 返回的稀疏矩阵。
3、r, c 是包含块的左上角的行坐标和列坐标的向量。
下面是一个基本的分离、聚合函数,说明 qtdecomp 的用法。
g = splitmerge(f, mindim, @predicate)
1、g 是输出图像,输出图像每个连接区域都使用不同的整数标注。
2、mindim 定义分解中允许的最小块尺寸,必须是 2 的正整数次幂。
flag = predicate(region)
如果在 region 区域中的像素满足函数中由代码定义的属性,那么函数就必须写成返回 true 的形式,否则,flag 的值就必须是false。
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH09\Fig0938(a)(cygnusloop_Xray_original).tif");
subplot(231),imshow(f),title('原始图像');
g32 = splitmerge(f,32,@predicate); %使用最小块为32进行分割
subplot(232),imshow(g32),title('使用最小块为32进行分割图像');
g16 = splitmerge(f,16,@predicate); %使用最小块为16进行分割
subplot(233),imshow(g16),title('使用最小块为16进行分割图像');
g8= splitmerge(f,8,@predicate); %使用最小块为8进行分割
subplot(234),imshow(g8),title('使用最小块为8进行分割图像');
g4 = splitmerge(f,4,@predicate); %使用最小块为4进行分割
subplot(235),imshow(g4),title('使用最小块为4进行分割图像');
g2 = splitmerge(f,2,@predicate); %使用最小块为2进行分割
subplot(236),imshow(g2),title('使用最小块为2进行分割图像');
图 (a)显示了一幅天鹅星座环的 X 射线频段图像。图像的大小为 256×256 像素。该例的目的是分割出环绕致密中心的稀疏环。
图 (b)到(f)显示了使用函数splitmerge且mindim的值分别等于 64、32、16、8、4 时分割图(a)的结果。所有图像均显示了边界的细节水平与 mindim 的值成反比的分割结果。
图中的所有结果都是合理的分割。如果以除原始图像之外的这些图像之一作为模板提取感兴趣区域,图(d)的结果将是最好的选择,因为它是具有最多细节的实心区域。
(五)使用分水岭变换的分割
分水岭变换将找到灰度图像中的集水盆地和脊线。集水盆地就是我们要识别的物体或区域。
5.1 使用距离变换的分水岭分割
二值图像的距离变换是每个像素到最接近非零像素的距离。
使用距离变换和分水岭变换分割二值图像 :
f = tofloat(imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH09\Fig0941(a)(wood_dowels).tif"));
g=im2bw(f,graythresh(f)); %把图像转换为二值图像
subplot(2,3,1),imshow(f);title('(a)使用距离和分水岭分割原图');
subplot(2,3,2),imshow(g);title('(b)原图像阈值处理后的图像');
gc=~g; %对图像求补
subplot(2,3,3),imshow(gc),title('(c)阈值处理后取反图像');
D=bwdist(gc); %计算其距离变换
subplot(2,3,4),imshow(D),title('(d)使用距离变换后的图像');
L=watershed(-D); %计算负距离变换的分水岭变换
w=L==0; %L 的零值即分水岭的脊线像素
subplot(2,3,5),imshow(w),title('(e)距离变换后的负分水岭图像');
g2=g & ~w; %原始二值图像和图像 w 的 “补” 的逻辑 “与” 操作可完成分割
subplot(2,3,6),imshow(g2),title('(f)阈值图像与分水岭图像相与图像');
使用im2bw和graythreshi把图像变换为二值图像,图 (a)显示了结果。
对图像求补,计算距离变换。然后,用函数watershed 计算距离变换的负分水岭变换。其中,L 是标记矩阵。A 是输入数组(一般可以是任何维数,但在本章是二维),并且conn指定了连通性(对于二维数组是 4 或 8(默认值))。
在L中,正整数与汇水盆地相对应,零值指出分水岭的脊线像素,图(b)和( c)显示了求补后的图像及其距离变换。因为L的 0 值像素是分水岭的脊线像素,图中仅显示这些像素。分水岭的脊线图像显示于图(d)中。
最后,使用原始的二值图像和图像w的“补”,通过逻辑 AND操作完成分割,如图 (e)所示,图(e)中的某些物体没有很好地分开。这被称为过分割,这是使用基于分水岭的分割方法时常常会出现的问题。
5.2 使用梯度的分水岭分割
在使用针对分割的分水岭变换之前,常常使用梯度幅度对图像进行预处理。梯度幅度图像沿着物体的边缘有较高的像素值,而在其他地方则有较低的像素值。在理想的情况下,分水岭变换可得到沿物体边缘的分水岭脊线。
使用梯度和分水岭变换分割灰度图像 :
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1056(a)(blob_original).tif");
subplot(221),imshow(f),title('原始图像');
h = fspecial('sobel');
fd = tofloat(f);
g = sqrt(imfilter(fd,h,'replicate').^2+imfilter(fd,h,'replicate').^2);
subplot(222),imshow(g,[]),title('梯度和分水岭分割幅度图像');
L =watershed(g);
wr = L == 0;
subplot(223),imshow(wr),title('严重分割过分割后图像');
g2 = imclose(imopen(g,ones(3,3)),ones(3,3));
L2 = watershed(g2);
wr2 = L2 == 0;
f2 = f;
f2(wr2) = 255;
subplot(224),imshow(f2),title('平滑梯度图像后的分水岭变换');
图( c)所示,这是不太好的分割结果;有太多的与我们感兴趣的物体边界不相对应的分水岭脊线。这是过分分割的另一个例子。针对这一问题的解决方法是在计算分水岭变换之前先平滑梯度图像。
图 (d)显示了叠加结果。虽然增强图©的目的达到了,但仍然存在一些附加的脊线,对于决定哪些汇水盆地真正与感兴趣物体有关还是很困难。
5.3 控制标记符的分水岭分割
由于噪声和梯度的其他局部不规则性,常会导致过分分割。一种方法是基于标记符修改梯度图像。标记符是属于一幅图像的连通分量。
标记符控制的分水岭分割示例 :
f = imread("C:\Users\madster\Downloads\DIP3E_Original_Images_CH10\Fig1057(a)(small_blobs-original).tif");
subplot(241),imshow(f),title('原始图像');
h = fspecial('sobel');
fd = tofloat(f);
g = sqrt(imfilter(fd,h,'replicate').^2+imfilter(fd,h,'replicate').^2);
L =watershed(g);
wr = L == 0;
subplot(242),imshow(wr),title('梯度幅度图像进行分水岭变换图像');
rm = imregionalmin(g); %计算图像中所有的局部小区域的位置
subplot(243),imshow(rm),title('梯度幅值的局部小区域图像');
im = imextendedmin(f,2); %用于计算比临近点更暗的图像中“低点”的集合
fim = f;
fim(im) = 175;
subplot(244),imshow(f,[]),title('内部标记符图像');
Lim = watershed(bwdist(im));
em = Lim == 0;
subplot(245),imshow(em,[]),title('外部标记符图像');
g2 = imimposemin(g,im|em); %用来修改一幅图像,使得其只在指定的位置处取得局部最小值
subplot(246),imshow(g2),title('修改后的梯度幅值图像');
L2 = watershed(g2);
f2 = f;
f2(L2 == 0) = 255;
subplot(247),imshow(f2),title('最后的分割图像');
在图(a)中可以看到,结果严重地过分割了。一部分是由于存在大量的小区域。
图(b)显示的多数局部小区域的位置非常浅,并且表现的细节与我们的分割问题不相干。
用函数imextendedmin来得到内部标记符的集合,最后将灰斑点的最小区域位置叠加在原始图像上,如图©所示。可以看到,得到的斑点的确很合理地标记了我们想要分割的物体。
图 (d)显示了二值图像em中的分水岭脊线。因为这些脊线位于由im标记的暗斑点 间,所以它们应该是很好的外部标记符。
通过在内部和外部标记符的位置覆盖局部最小区域,可以改进梯度图像,图(e)显示了结果。
在原始图像上叠加分水岭脊线,得到一个有很大改进的分割结果如图(f)所示。
相关链接:
(https://blog.csdn.net/Dujing2019/article/details/90203492)
相关链接:
(https://blog.csdn.net/mary_0830/article/details/89597672)