Is there a way to eliminate the disconnected elements of an image?

134 Views Asked by At

I'm working on surface roughness of an image, and I used the edge detection function from opencv, but it also detected the pores from the image. Is there a way to get rid of the pores?

picture of original image and the edges detected

I tried adjusting the threshold values but that only got rid of the edges on the side and made the pores more pronounced. I´m pretty new to opencv and this type of problem, so most of this code is just based on examples I saw online.

import numpy as np
import cv2 as cv
from matplotlib import pyplot as plt
from scipy import ndimage

# need to add the scale to it
img = cv.imread('testim5.png', cv.IMREAD_GRAYSCALE)
blurred = ndimage.filters.gaussian_filter(img, 2, mode='nearest')
edges = cv.Canny(blurred,100,200) # adjusting thresholds will only eliminate the outside edges, not the pores 

[testim5.png] https://i.stack.imgur.com/Vs6K8.png

1

There are 1 best solutions below

1
On BEST ANSWER

First, you could use morphological operations to ensure that the interesting edges are connected. You can read more about morphological operations in the web of OpenCV: https://docs.opencv.org/4.x/d9/d61/tutorial_py_morphological_ops.html.

The code would be:

import numpy as np
import cv2 as cv
import matplotlib.pyplot as plt
from scipy import ndimage

# Open the image
img = cv.imread('testim5.png', cv.IMREAD_GRAYSCALE)

# Note that `scipy.ndimage.filters.gaussian_filter` is deprecated, use `scipy.ndimage.gaussian_filter` instead
blurred = ndimage.gaussian_filter(img, 2, mode='nearest')
edges = cv.Canny(blurred,100,175)

# Apply morphological operation (closing) to the EDGES so they are connected
kernel = np.ones((5,3),np.uint8)
edges_closing = cv.morphologyEx(edges, cv.MORPH_CLOSE, kernel)

The result of this is that the vertical edges aren't so disjoint, as you can see in the following image.

Figure 1

The second step is to remove the pores. It can be seen as a problem of removing small connected objects and the solution of @Soltius in How to remove small connected objects using OpenCV works well.

The code in your case would be:

# Remove smallest connected components
# To understand the following lines, look the answer of @Soltius in https://stackoverflow.com/questions/42798659/how-to-remove-small-connected-objects-using-opencv
nb_blobs, im_with_separated_blobs, stats, _ = cv.connectedComponentsWithStats(edges_closing)
sizes = stats[:, -1]
sizes = sizes[1:]
nb_blobs -= 1

# Minimum size of particles we want to keep (number of pixels).
# Here, it's a fixed value, but you can set it as you want.
min_size = 80  

final_edges = np.zeros_like(im_with_separated_blobs)
for blob in range(nb_blobs):
    if sizes[blob] >= min_size:
        final_edges[im_with_separated_blobs == blob + 1] = 255

fig, axs = plt.subplots(ncols=3)
axs[0].imshow(img, cmap='gray')
axs[1].imshow(edges, cmap='gray')
axs[2].imshow(final_edges, cmap='gray')
axs[0].set_title('Original image')
axs[1].set_title('Original edges')
axs[2].set_title('New edges')
fig.tight_layout()
plt.show()

The final edges are:

Figure 2