图像相似度算法
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
1、图片大小规格化
为了比较两个图像,应该使其大小完全一致,这里可以设置为两个图片中较大的那个大小,长宽。
function Resize(const Source: TBitmap; var Dest: TBitmap): Boolean;
begin
if not Assigned(Dest) then
Dest := TBitmap.Create;
Dest.pixelformat := pf24bit;
Dest.Width := BMPWIDTH;
Dest.Height := BMPHEIGHT;
Dest.Canvas.CopyRect(Rect(0, 0, Dest.Width - 1, Dest.Height - 1), Source.Canvas, Rect(0, 0, Source.Width - 1, Source.Height - 1));
end;
2、图像灰度化
图像灰度化的方法有多种,这里介绍两种。一种是绝对平均值,一种是加权平均值。其实质就是将RGB三原色的色值相加,平均后赋予新值。不同的只是RGB三原色的权重不同。
绝对平均值
function Gray1(const Source: TBitmap): Boolean;
var
p : PByteArray;
w : Integer;
i, j : Integer;
begin
for i := 0 to Source.Height - 1 do
begin
p := Source.ScanLine[i];
for j := 0 to (Source.Width - 1) do
begin
w := p[3 * j] + p[3 * j + 1] + p[3 * j + 2];
w := w div 3;
w := byte(w);
p[3 * j] := w;
p[3 * j + 1] := w;
p[3 * j + 2] := w;
end;
end;
end;
加权平均值
function TForm1.Gray2(const Source: TBitmap): Boolean;
var
p : PByteArray;
w : Integer;
i, j : Integer;
begin
for i := 0 to Source.Height - 1 do
begin
p := Source.ScanLine[i];
for j := 0 to (Source.Width - 1) do
begin
w := (p[3 * j] * 28 + p[3 * j + 1] * 151 + p[3 * j + 2] * 77);
w := w div 3;
w := byte(w);
p[3 * j] := w;
p[3 * j + 1] := w;
p[3 * j + 2] := w;
end;
end;
end;
3、抽取直方图
由于是灰度图,所以只会有256种灰度值,计算直方图
function GetHisogram(const Source: TBitmap; var His: TSamp): Boolean; var
i, j, c : Integer;
p : PByteArray;
begin
try
//取样
for i := 0 to 255 do
His[i] := 0;
for i := 0 to Source.Height - 1 do
begin
p := Source.ScanLine[i];
for j := 0 to Source.Width - 1 do
begin
c := p^[j * 3];
Inc(His[c]);
end;
end;
finally
tb.Free;
end;
end;
4、判断相似度
这里采用的是一个公式,,g,s分别为两附图片的直方图,N为直方图抽样个数。
function GetSimilar(const HisA, HisB: TSamp): Double;
var
i, j : Integer;
function DivideAbs(const NumA, NumB: Integer): Double;
var
Tabs : Integer;
begin
Tabs := Abs(NumA - NumB);
if NumA >= NumB then
Result := NumA
else
Result := NumB;
if Result <> 0 then
Result := Tabs / Result;
end;
begin
Result := 0.0;
if Length(HisA) <> Length(HisB) then
Exit;
for i := 0 to Length(HisA) do
begin
Result := Result + 1 - DivideAbs(HisA[i], HisB[i]); end;
Result := Result / Length(HisA);
end;