How to perform batch image thresholding with variable threshold for each image?

7 vues (au cours des 30 derniers jours)
I have a 3d matrix of grayscale voxels that I am attempting to 'slice' into a stack of 2D images, and then perform image segmentation on each image, then recombine the result.
The problem I am having is that in the histogram, the intensity of the desired section (bone) appears to vary from image to image. Global threshold doesn't seem to work -- it includes the skin. Manual threshold works for images similar to the image in I use as the basis for determining the threshold, but fails on others. Adaptive threshold includes many portions outside the region I want, and not all the ones inside. I've also tried triangle thresholding, and hysteresis thresholding (I've had some degree of success with this but it's less than ideal.)
There are two peaks in the histogram, the dark area of empty space surrounding the subject, and the subject, respectively. The portion of the image where the intensity is beyond the second peak is the information I want.
Is anyone aware of an image thresholding technique that would enable me to output only these areas of the image?
Example (in the first image I'd want the portion where intensity > 0.425, the second > 0.5):
Image 59
Image 106

Réponse acceptée

Erik Johnson
Erik Johnson le 24 Mai 2016
Modifié(e) : Erik Johnson le 24 Mai 2016
I figured out how to do this based on the second peak in the histogram. It feels kind of hackish and unwieldy, but it works.
Assuming MZ is your two dimensional image:
hist = imhist(MZ);
% Get counts and bins for histogram
[counts, binlocs] = imhist(MZ);
% Find peaks of histogram
[peaks,locs] = findpeaks(hist);
% Get 2nd peak by sorting in descending order and choosing second index
peaks_sorted = sort(peaks,'descend');
second_peak = peaks_sorted(2);
%Find index of 2nd max
second_peak_index = find(counts==second_peak);
second_peak_index = max(second_peak_index);
% If multiple instances of intensity of second_peak exist, we want the
% highest index. There is probably a better way to do this.
%Determine intensity value of second peak
second_peak_intensity = binlocs(second_peak_index);
%Choose threshold value based on second peak
thresh_value = second_peak_intensity + 0.01;
%Create image segmentation based on dynamic threshold value
BW = MZ > thresh_value;
seg = MZ;
seg(~BW) = 0;

Plus de réponses (1)

Image Analyst
Image Analyst le 22 Mai 2016
Modifié(e) : Image Analyst le 22 Mai 2016
The standard Otsu method in graythresh works well for these kinds of high contrast, bimodal images to find thresholds between the humps. However you don't want a threshold between the humps. Triangle thresholding would be perfect for getting the thresholds you want. Maybe you're not using a good implementation of it. I wrote my own. Please upload an image so I can use it.
  8 commentaires
Image Analyst
Image Analyst le 24 Mai 2016
Well you can just crop that out since I'm sure it's a constant. I was going to show you my triangle threshold method but it looks like you've accepted an answer that works for you. Anyway good luck with the rest of your project.
Erik Johnson
Erik Johnson le 25 Mai 2016
Sorry, I didn't see a response from you for a few days so I went ahead and accepted my own answer. I would still like to see the triangle threshold method, if you can show me. The variable threshold method works to some degree, but the 'fudge factor' is still static and requires manual intervention, which is less than ideal for my use case.

Connectez-vous pour commenter.

Catégories

En savoir plus sur Image Segmentation and Analysis dans Help Center et File Exchange

Produits

Community Treasure Hunt

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

Start Hunting!

Translated by