Kachelung von Bildern - der naive Ansatz#

Bei der gekachelten Bildverarbeitung besteht der erste Schritt darin, das Bild in Kacheln zu zerschneiden. Während dies mit numpy gemacht werden könnte, werden wir dask verwenden, da es in diesem Zusammenhang mehrere sehr nützliche Funktionen mitbringt.

import dask
import dask.array as da
from skimage.filters import gaussian
from skimage.data import cells3d
from pyclesperanto_prototype import imshow

Im ersten Beispiel werden wir ein Bild verwenden, das Zellkerne aus der Fluoreszenzmikroskopie zeigt, und es einfach mit einem Gauß-Filter entrauschen. Wir werden dies Kachel für Kachel durchführen. Dafür definieren wir zunächst die procedure, die auf alle Kacheln angewendet werden soll. Wir bauen in diese Funktion eine print-Anweisung ein, um zu sehen, wann sie ausgeführt wird und wie groß das Bild ist, das verarbeitet wird.

def procedure(image):
    print("proceduring", image.shape)
    return gaussian(image, sigma=5)
image = cells3d()[30,1]
imshow(image)
../_images/e6d2393f7c605c769fb82b772552ade1566917e2a294f57aa9e4f747b0c78793.png

Nach dem Laden des Bildes kann es wie folgt gekachelt werden. In dask werden Kacheln auch als chunks bezeichnet.

tiles = da.from_array(image, chunks=(128, 128))
tiles
Array Chunk
Bytes 128.00 kiB 32.00 kiB
Shape (256, 256) (128, 128)
Count 4 Tasks 4 Chunks
Type uint16 numpy.ndarray
256 256

Als Nächstes teilen wir dask mit, was mit unseren Kacheln geschehen soll. Wir wollen die Funktion procedure auf alle einzelnen Kacheln mappen. Beachten Sie, dass dies das gesamte Bild noch nicht verarbeitet.

tile_map = da.map_blocks(procedure, tiles)
proceduring (0, 0)
proceduring (1, 1)

Wie wir lesen können, wurde die Funktion zweimal mit sehr kleinen Bildern (0x0 und 1x1 Pixel) ausgeführt. Dask macht das prinzipiell, um zu erkunden, ob die Funktion funktioniert. Als Nächstes werden wir unsere procedure tatsächlich auf den Kacheln des Bildes ausführen.

result = tile_map.compute() # Warnung: Dies lädt alle Bilddaten in den Speicher
proceduringproceduring (128, 128)
 (128, 128)
proceduring (128, 128)
proceduring (128, 128)

Die gedruckte Ausgabe sieht etwas chaotisch aus, weil dask die procedure parallel auf mehreren Kacheln ausgeführt hat. Wenn wir das Ergebnis untersuchen, werden wir sehen, dass es wieder ein Bild ist.

result.shape
(256, 256)
type(result)
numpy.ndarray

Hinweis: Die imshow-Funktion funktioniert möglicherweise nicht bei großen Datensätzen. Wir verwenden sie hier zu Demonstrationszwecken.

imshow(result)
../_images/7b0a8ba73291ddde763b05183cfdf2a3d3557bede55f4d5dd6dd35a5022f1c2e.png

Randeffekte#

Bei der Verarbeitung von Bildern Kachel für Kachel müssen wir immer davon ausgehen, dass entlang der Ränder Artefakte auftreten, die aus dem Zerschneiden des Bildes in Kacheln resultieren. Da unser Beispielbild in den Speicher passt, können wir procedure darauf anwenden und es mit dem Ergebnis der gekachelten Bildverarbeitung vergleichen.

untiled_result = procedure(image)
imshow(untiled_result)
proceduring (256, 256)
../_images/b31924a9f3cf423d048676a3c5ccca6648eb37f59f08ecc030e068593386819d.png

Die Unterschiede sind nicht offensichtlich, aber wir können sie visualisieren.

difference = result - untiled_result
imshow(difference)
../_images/e4126918c50e4b551767159a05ace9b17cb60db4891f34ce4000369b3cfe9c54.png

Bei Anwendung eines Gauß-Filters mit einem kleinen Sigma können diese Effekte vernachlässigbar sein. Falls die Effekte schwerwiegende Probleme in unserem Bildverarbeitungs-Workflow verursachen, möchten wir diese Artefakte möglicherweise reduzieren oder sogar verhindern.