Segmentacja płuc MatLab: 5 kroków
Segmentacja płuc MatLab: 5 kroków
Anonim
Segmentacja płuc MatLab
Segmentacja płuc MatLab

Autor: Phuc Lam, Paul Yeung, Eric Reyes

Uznanie, że błędy w segmentacji płuc dadzą fałszywe informacje dotyczące identyfikacji obszaru chorobowego i mogą bezpośrednio wpłynąć na proces diagnozy. Nowoczesne techniki wspomagania komputerowego nie zapewniały dokładnych wyników, gdy choroby płuc mają trudne kształty. Te nieprawidłowe kształty mogą być spowodowane wysiękiem opłucnowym, konsolidacją itp. Stosując technikę segmentacji płuc, w której granice płuc są izolowane od otaczającej tkanki klatki piersiowej, nasza aplikacja może zidentyfikować granice z progami wejściowymi użytkownika, aby uzyskać w pełni konfigurowalne widoki kształtów płuc, Celem tego projektu MatLab jest stworzenie przyjaznej dla użytkownika interaktywnej aplikacji do segmentacji płuc w celu wykrywania patologicznych stanów na zdjęciach rentgenowskich płuc. Naszym celem jest stworzenie bardziej skutecznego sposobu ilustrowania i identyfikowania nieprawidłowych płuc, aby dać lekarzom i radiologom bardziej niezawodny sposób diagnozowania chorób płuc. Korzystając z narzędzia do projektowania aplikacji w MatLab, program jest przeznaczony do pracy w szczególności ze skanami RTG klatki piersiowej i tomografii komputerowej (CT), ale jest również testowany pod kątem pracy ze skanami MRI.

Poniższe instrukcje zawierają naszą technikę filtrowania szumu (dolnoprzepustowy filtr Wienera), a także próg obrazu (przy użyciu histogramu intensywności obrazu w skali szarości) oraz przy użyciu gradientu morfologicznego (różnica między dylatacją a erozją obrazu) do zidentyfikować region zainteresowania. Instrukcja wyjaśni następnie, w jaki sposób integrujemy wszystkie elementy w graficznym interfejsie użytkownika (GUI).

Notatka:

1). Ten projekt jest inspirowany artykułem badawczym: „Segmentacja i analiza obrazu nieprawidłowych płuc w tomografii komputerowej: obecne podejścia, wyzwania i przyszłe trendy”. Które można znaleźć tutaj

2). Korzystamy ze zdjęć rentgenowskich z NIH: Clinical Center. Link można znaleźć tutaj

3). Pomoc dla projektanta aplikacji można znaleźć tutaj

4). Przed uruchomieniem kodu: należy zmienić ścieżkę Dir (w linii 34) na katalog plików i typ obrazu (w linii 35) (analizujemy *.png).

Krok 1: Krok 1: Ładowanie obrazu

Krok 1: Ładowanie obrazu
Krok 1: Ładowanie obrazu

Ten krok pokaże oryginalny obraz w skali szarości. Zmień 'name_of_picture.png' na nazwę swojego obrazu

jasne; kl; zamknij wszystko;

%% Ładowanie obrazów

raw_x_ray='nazwa_obrazu.png';

I=imread(raw_x_ray);

figura(101);

imshow(I);

mapa kolorów (szary);

title('Rentgen w skali szarości');

Krok 2: Krok 2: Filtrowanie szumów i histogram

Krok 2: Filtrowanie szumów i histogram
Krok 2: Filtrowanie szumów i histogram

Aby znaleźć próg dla obrazu w skali szarości, patrzymy na histogram, aby zobaczyć, czy istnieją różne tryby. Przeczytaj więcej tutaj

I=wiener2(I, [5 5]);

figura(102);

działka podrzędna(2, 1, 1);

imshow(I);

działka podrzędna(2, 1, 2);

imista (I, 256);

Krok 3: Krok 3: Ustawianie progów

Krok 3: Ustawianie progów
Krok 3: Ustawianie progów
Krok 3: Ustawianie progów
Krok 3: Ustawianie progów

Ten krok pozwala ustawić próg zgodnie z histogramem. morphologicalGradient podświetli interesujący obszar na czerwono, a funkcja visboundaries nałoży obrysowany i przefiltrowany obraz płuca na czerwono.

Używając regionprops, możemy określić tablice solidności i posortować je malejąco. Następnie zbinaryzuję obraz grey sclae i zastosuję metodę gradientu morfologicznego oraz mLoren Shurasking, aby podświetlić interesujący obszar (ROI). Następnym krokiem jest odwrócenie obrazu, tak aby obszar ROI płuc był biały na czarnym tle. Używam funkcji showMaskAsOverlay do wyświetlenia 2 masek. Uwaga: kod jest inspirowany przez Loren Shure, link.

Na koniec tworzę czerwony kontur, używając bwbwboundaries i maskując obraz filtra i granice.

a_thresh = I >= 172; % ustaw ten próg

[labelImage, numberOfBlobs] = bwlabel(a_thresh);

props = regionprops(a_thresh, 'all');

sortedSolidity = sort([props. Solidity], 'descend');

SB = posortowanaSolidność(1);

jeśli SB == 1 % SB akceptuje tylko solidność == 1 odfiltrowuje kości

obraz binarny = imbinaryzacja(I); figura(103);

imshow(obraz binarny); mapa kolorów (szary);

SE = strel('kwadrat', 3);

Gradient morfologiczny = imsubtract(imdilate(binaryImage, SE), imerode(binaryImage, SE));

mask = imbinarize(Gradient morfologiczny, 0,03);

SE = strel('kwadrat', 2);

maska = imclose(maska, SE);

mask = imfill(maska, 'dziury');

maska = bwareafilt(maska, 2); % kontrolny numer obszaru pokaż

notMask = ~maska;

maska = maska | bwpropfilt(notMask, 'Obszar', [-Inf, 5000 - eps(5000)]);

showMaskAsOverlay(0.5, maska, 'r'); % musisz pobrać aplikację/funkcję showMaskAsOverlay

BW2 = imfill(binaryImage, 'dziury');

nowy_obraz = BW2;

nowy_obraz(~maska) = 0; % odwrócone tło i dziury

B=bwgranice(nowy_obraz); % może zaakceptować tylko 2 wymiary

figura(104);

imshow(nowy_obraz);

trzymać się

visgranice(B);

kończyć się

Krok 4: Tworzenie GUI

Teraz integrujemy poprzedni kod z aplikacją MATLAB. Otwórz Projektanta aplikacji w MATLAB (Nowy > Aplikacja). Najpierw projektujemy interfejs, przytrzymując i przeciągając w trzech osiach do środkowej przestrzeni roboczej. Następnie klikamy-przytrzymaj-przeciągnij dwa przyciski, jedno pole edycyjne (tekst), jedno pole edycyjne (numeryczne), jeden suwak i jedno menu rozwijane. Każda z dwóch osi wyświetli podgląd i przeanalizowany obraz, a trzecia oś wyświetli histogram pikseli dla „wybranego” obrazu podglądu. Pole edycyjne (tekstowe) wyświetli ścieżkę pliku wybranego obrazu, a pole edycyjne (numeryczne) wyświetli wykryty obszar pikseli w płucach.

Teraz przełącz się z widoku projektu na widok kodu w Projektancie aplikacji. Wprowadź w kodzie kod nieruchomości, klikając czerwony przycisk „Właściwości” ze znakiem plus. Zainicjuj właściwości I, threshold i regionsToExtract, jak w kodzie podanym poniżej. Następnie kliknij prawym przyciskiem myszy przycisk w prawym górnym rogu obszaru roboczego (Przeglądarka komponentów) i przejdź z Callbacks>Go to… callback. Dodaj kod dla „funkcji SelectImageButtonPushed(aplikacja, zdarzenie)”. Ten kod pozwala wybrać obraz do analizy z komputera za pomocą uigetfile. Po wybraniu obrazu pod osiami pojawi się obraz podglądu wraz z histogramem. Następnie kliknij prawym przyciskiem myszy drugi przycisk i powtórz tę samą procedurę, aby utworzyć funkcję zwrotną.

Dodaj kod poniżej „funkcja AnalyzeImageButtonPushed(app, event).” Ten kod wykona zliczanie pikseli i wykrywanie blobów na obrazie podglądu po naciśnięciu przycisku analizy obrazu (w zależności od tego, który został kliknięty prawym przyciskiem myszy dla tego kodu). Po zaprogramowaniu przycisków zaprogramujemy teraz suwak i rozwijane menu. Kliknij suwak prawym przyciskiem myszy, utwórz funkcję wywołania zwrotnego i dodaj kod poniżej „function FilterThresholdSliderValueChanged(app, event)” do końca. Pozwala to suwakowi dostosować próg intensywności szarości.

Utwórz funkcję wywołania zwrotnego dla menu rozwijanego i dodaj kod poniżej „funkcja AreastoExtractDropDownValueChanged(app, event)”, aby menu rozwijane mogło modyfikować liczbę obiektów blob wyświetlanych na analizowanych osiach obrazu. Teraz kliknij każdy element w Przeglądarce komponentów i zmień jego właściwości według własnych upodobań, takie jak zmiana nazw elementów, usunięcie osi i zmiana skalowania. Przeciągnij i upuść elementy przeglądarki komponentów w widoku projektu do funkcjonalnego i łatwego do zrozumienia układu. Masz teraz aplikację w MATLAB, która może analizować obrazy płuc pod kątem obszaru pikseli!

właściwości (dostęp = prywatny)I = ; % plik graficzny

próg = 257; %próg dla binaryzacji intensywności szarości

regionyDoWyciągu = 2;

kończyć się

funkcja SelectImageButtonPush (aplikacja, zdarzenie)

clc;Dir = 'C:\Użytkownicy\danie\Pobrane\images_004\images'; %define plik niezmienny "prefiks"

[imageExt, path] = uigetfile('*.png'); % chwyć zmienną część nazwy obrazu

imageName = [Dir fileep imageExt]; %konkatenacja szczepów niezmiennych i zmiennych

app. I = imread(nazwaobrazu); %przeczytaj obrazek

imshow(app. I, 'rodzic', app. UIAxes); % wyświetl obraz

app. FilePathEditField. Value = ścieżka; %wyświetlana ścieżka pliku, z którego pochodzi oryginalny obraz

kończyć się

funkcja AnalyzeImageButtonPush (aplikacja, zdarzenie)

obraz oryginalny = aplikacja. I;

obraz oryginalny = wiener2(app. I, [5 5]); %filtr usuwania kropek

histogram (app. AxesHistogram, app. I, 256); % wyświetl histogram obrazu

a_thresh = obraz oryginalny >= app.threshold; % ustaw ten próg

labelImage = bwlabel(a_thresh);

props = regionprops(a_thresh, 'all');

sortedSolidity = sort([props. Solidity], 'descend');

SB = posortowanaSolidność(1);

jeśli SB == 1 % SB akceptuje tylko solidność ==1 odfiltrowuje kości

SE = strel('kwadrat', 3);

morphologicalGradient = imsubtract(imdilate(labelImage, SE), imerode(labelImage, SE));

mask = imbinarize(Gradient morfologiczny, 0,03);

SE = strel('kwadrat', 2);

maska = imclose(maska, SE);

maska = imfill(maska, 'dziury');

maska = bwareafilt(maska, app.regionsToExtract);

% kontrolny numer obszaru pokaż

notMask = ~maska;

maska = maska | bwpropfilt(notMask, 'Obszar', [-Inf, 5000 - eps(5000)]);

BW2 = imfill(labelImage, 'dziury');

nowy_obraz = BW2;

nowy_obraz(~maska) = 0;

B = bwboundaries(nowy_obraz); % może zaakceptować tylko 2 wymiary imshow(new_image, 'parent', app. UIAxes2);

hold(app. UIAxes2, 'on');

visgranice(B);

set(gca, 'YDir', 'reverse');

lungArea = bwarea(nowy_obraz);

app. PixelAreaEditField. Value = lungArea;

kończyć się

kończyć się

funkcja FilterThresholdSliderValueChanged(aplikacja, zdarzenie)

app.threshold = app. FilterThresholdSlider. Value;

kończyć się

function AreastoExtractDropDownValueChanged(app, event)stringNumber = app. AreastoExtractDropDown. Value;

app.regionsToExtract = str2double(stringNumber);

kończyć się

kończyć się