大津算法详解

一、算法功能

图像分割就是把图像分成若干个特定的、具有独特性质的区域并提出感兴趣目标的技术和过程。它是由图像处理到图像分析的关键步骤。

大津算法也称最大类间差法,由大津于1979年提出,被认为是图像分割中阈值选取的最佳算法,计算简单,不受图像亮度和对比度的影响,因此在数字图像处理上得到了广泛的应用。

二、图像的简单分类

一般而言,图像分为彩色图,灰度图,二值图。在计算机处理中,一般将图片作为一个包含多个通道的像素矩阵来处理。若将图片长度计为m,图片宽度计为n,有如下:

彩色图:有R,G,B三原色。三个通道,每个通道的矩阵大小为图片大小。矩阵大小为m×n×3

灰度图:只有一个颜色通道,值域为0-255,矩阵大小为m×n

二值图:只有一个颜色通道,值域为0或1,矩阵大小为m×n

三、大津分割算法

大津算法目标为从所有的像素值中寻找类间方差最大的像素值,将其作为分割的界限,即阈值。使得比该阈值小的像素值为0,其余为1。

1.类间方差
ω0=N0/ (M×N)             (1)
      ω1=N1/ (M×N)             (2)
    N0+N1=M×N                (3)
    ω0+ω1=1                (4)
    μ=ω0*μ0+ω1*μ1            (5)
    g=ω0(μ0-μ)^2+ω1(μ1-μ)^2  (6)
将式(5)代入式(6),得到等价公式:
    g=ω0ω1(μ0-μ1)^2              %类间方差方差
2.求类间方差最大值对应的像素值

首先,从图片的像素点中寻找出像素最大值与像素最小值。我们要遍历的灰度值便在这最大值与最小值之间。

Max = max(Imag(:));               % 最大值
Min = min(Imag(:));               % 最小值
T = Min:Max;                     % 灰度值范围

其次,遍历这些灰度值,按照公式计算前景比例,背景比例,前景灰度平均值,背景灰度平均值。并按照这些参数计算方差。

for i = 1 : length(T)
TK = T(i);
iFg = 0;          % 前景
iBg = 0;          % 背景
FgSum = 0;    % 前景总数
BgSum = 0;    % 背景总数
W1=Imag>TK;
WK1=W1.*Imag;
iFg1=sum(W1(:));%求前景比例
iBg1=imagSize-iFg1;
FgSum1=sum(WK1(:));
BgSum1=sum(Imag(:))-FgSum1;

w0 = iFg1/imagSize;      % 前景比例
w1 = iBg1/imagSize;     % 背景比例
u0 = FgSum1/iFg1;         % 前景灰度平均值
u1 = BgSum1/iBg1;        % 背景灰度平均值
Tval(i) = w0*w1*(u0 - u1)*(u0 - u1);     % 计算方差
end

得到方差矩阵后,寻找最大下标对应的像素值即为阈值。(部分代码参考(52条消息) MATLAB图像处理——大津阈值分割(附代码)_HNU_刘yuan的博客-大津阈值,对其进行了优化)

[~, flag] = max(Tval);             % 最大值下标
TValue = T(flag);
3.根据得到的阈值进行图像分割
BW = im2bw(I, T/255);%转为二值图像
4.结果展示

opencv大津阈值算法 大津算法matlab_方差

四、完整代码

clear; clc;
F=imread('SF.jpg');%彩图
I=rgb2gray(F);%灰度图
subplot(2, 2, 1)
imshow(I);
xlabel('(a) 灰度图');

T = OTSU(double(I));     %使用大津法计算阈值
disp(['大津法计算灰度阈值:', num2str(T)])
BW = im2bw(I, T/255);%转为二值图像
%阈值分割
subplot(2, 2, 2)
imshow(BW);
xlabel('(b) 大津法');
 
subplot(2, 2, 3)
imshow(F);
xlabel('(c) 彩色图');
function TValue = OTSU(Imag)

Max = max(Imag(:));              % 最大值
Min = min(Imag(:));               % 最小值
T =Min:Max;                        % 灰度值范围
Tval = zeros(size(T));               % 方差


[Row,Col]=size(Imag);


imagSize = Row*Col;            % 像素点数量
% 遍历灰度值
for i = 1 : length(T)
    TK = T(i);
    iFg = 0;          % 前景
    iBg = 0;          % 背景
    FgSum = 0;    % 前景总数
    BgSum = 0;    % 背景总数
    W1=Imag>TK;
    WK1=W1.*Imag;
    iFg1=sum(W1(:));%求前景比例
    iBg1=imagSize-iFg1;
    FgSum1=sum(WK1(:));
    BgSum1=sum(Imag(:))-FgSum1;

    w0 = iFg1/imagSize;      % 前景比例
    w1 = iBg1/imagSize;     % 背景比例
    u0 = FgSum1/iFg1;         % 前景灰度平均值
    u1 = BgSum1/iBg1;        % 背景灰度平均值
    Tval(i) = w0*w1*(u0 - u1)*(u0 - u1);     % 计算方差
end
[~, flag] = max(Tval);             % 最大值下标
TValue = T(flag);
end