I have a grayscale (value 0-255) image with background pixel equals 0. I am trying to remove noises at the border between the foreground and the background. These noises can be identified by the big jump in values. Is there any existing algorithm/filter that can achieve this? If no, may I know the most efficient way to do it instead of comparing pixels with its neighborhood?
The result that I expect is to removal of those noises by changing them to background pixel (0) without changing any of the border/foreground pixel values.

 Réponse acceptée

Walter Roberson
Walter Roberson le 9 Jan 2016

1 vote

bigjump = diff(YourArray(:,1:end-1),1,1) > 32 | diff(YourArray(1:end-1,:),1,2) > 32);
bigjump(end+1,end+1) = false;
diff shortens the array in one direction but not the other, so to make arrays that are compatible to be or'd together, I take one fewer row or column in the other direction. Then the line after that is to pad out the matrix to the original size.
Now if you want to create mask from the first occurrence onward, you need to a cumulative "or". That can be done by cumsum() the logical value and comparing the result to 1 -- at least 1 true value at or before a location will result in 1 or more at that location.
jumped = cumsum(bigjump, 1) >= 1 | cumsum(bigjump, 2) >= 1;
and then
NewArray = YourArray;
NewArray(jumped) = 0;

Plus de réponses (1)

Image Analyst
Image Analyst le 9 Jan 2016

1 vote

An alternative way is to use mathematical morphology. You can get the local min in a 3x3 (or whatever size or shape you want) using imerode()
localMin = imerode(grayImage, true(3));
% Find jumps by subtracting from the original
jumpThreshold = 32; % or whatever you want.
jumps = (grayImage - localMin) > jumpThreshold ; % This is a mask!
% Set big jump pixels to zero.
grayImage(jumps) = 0;
You could boil all that down to 2 lines of code (or even 1) if you wanted it to be super compact.

6 commentaires

Hg
Hg le 9 Jan 2016
Great! This is a good alternative. Now I'm going to find out whether there's a way to remove the noises without having to set the jumpThreshold manually.
Image Analyst
Image Analyst le 9 Jan 2016
Well, there's the median filter, medfilt2() - maybe you'll like what that does. It replaces the center pixel in a moving window by the median of pixel values in the window. Why do you get the boost at the edge anyway? Maybe your enhancement scheme could be better? Why do you need to clip/zero/remove the bright edges anyway? What is your overall goal?
Hg
Hg le 9 Jan 2016
Modifié(e) : Hg le 9 Jan 2016
Those pixel values are actually the distance (depth) from the camera so I need to keep those values as they are. Those 'bright' edges are noises that need to be removed. The background pixels should have large values but I changed it to 0 for convenience.
Why not just threshold
distanceImage(distanceImage>200) = inf;
Hg
Hg le 9 Jan 2016
Modifié(e) : Hg le 9 Jan 2016
I can't do that as some parts of the foreground have values close to or larger than 200. So the way I define noises is through the big jump in values.
Image Analyst
Image Analyst le 9 Jan 2016
A slight variant of what I code I gave is to use imopen(). It does an imerode() (local min) followed by imdilate() (local max). Try that also and see how it looks. Try varying the kernel size.

Connectez-vous pour commenter.

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!

Translated by