• 【opencv】形态学重建案例-数糖果(细胞)个数


     原始图片如下,要求是利用形态学重建方法数出糖果个数

    原始图片:a.jpg
    ​​​​

    step1:先进行测地膨胀

    令F表示标记图像,令G表示模板图像。在讨论中,我们假设两幅图像都是二值图像,且F包含于G。标记图像相对于模板大小为1的测地膨胀定义为

    F相对于G的大小为n的测地膨胀定义为:

    式中,n≥1是整数,D0=F。在这个递归公式中,每一步都是膨胀后取交集。交集运算可以保证模板G限制标记F的生长(膨胀)。下图是大小为1的测地膨胀。

    图1:迭代曲线

     

    图2:测地膨胀后的结果

     step2:进行画面去噪声 高斯模糊等操作,使得连通域的数量与糖果数量接近

    图3: 使得连通域的数量与糖果数量接近

    step3:标记连通域数量,得出糖果数 

    图4:最终结果

     完整代码

    1. import cv2
    2. import numpy as np
    3. import matplotlib.pyplot as plt
    4. origin = cv2.imread("a.jpg")
    5. imgGray = cv2.imread("a.jpg", flags=0) # flags=0 灰度图像
    6. ret, imgBin = cv2.threshold(imgGray, 205, 255,
    7. cv2.THRESH_BINARY) # 二值化处理 (黑色0/白色1) ret:阈值 205:阈值 255:最大值 cv2.THRESH_BINARY:二值化
    8. imgMask = cv2.bitwise_not(imgBin) # 二值图像的补集 (黑色1/白色0)
    9. # 构造标记图像
    10. marker = np.zeros_like(imgBin, dtype=np.uint8) # 与imgBin同尺寸的全0图像
    11. marker[0, :] = imgBin[0, :] # 第一行
    12. marker[-1, :] = imgBin[-1, :] # 最后一行
    13. marker[:, 0] = imgBin[:, 0] # 第一列
    14. marker[:, -1] = imgBin[:, -1] # 最后一列
    15. markerIni = marker.copy() # 标记图像: 边框 f(x,y)=I(x,y),其它为 0
    16. # 形态学重建
    17. element = cv2.getStructuringElement(cv2.MORPH_CROSS, (3, 3))
    18. # 创建数组画出迭代图
    19. diffList = []
    20. while True: # 进行测地膨胀 是有条件的膨胀
    21. marker_pre = marker # 保存 F(n-1)
    22. dilation = cv2.dilate(marker, kernel=element) # 膨胀重建 目的是扩大 F(n-1) 的边界
    23. marker = cv2.bitwise_and(dilation, imgMask) # 原图像作为模板用来约束重建,按位与,有 0 得 0 目的是保持 F(n-1) 的内部
    24. # 计算重建后的标记图像与重建前的标记图像的差异
    25. diff = cv2.subtract(marker, marker_pre)
    26. print(diff.sum())
    27. diffList.append(diff.sum())
    28. if (marker_pre == marker).all(): # F(n)=F(n-1)?,判断是否达到稳定收敛状态
    29. break
    30. imgRebuild = cv2.bitwise_or(imgBin, marker) # 按位或,有 1 得 1
    31. # 绘制迭代图,命名为"图1:多次迭代的图像差异"
    32. plt.figure("图1:多次迭代的图像差异")
    33. plt.plot(diffList)
    34. plt.show()
    35. # 显示
    36. # cv2.imshow("imgGray", imgGray)
    37. # cv2.imshow("imgBin", imgBin)
    38. cv2.imshow("figure2:after Expansion", imgMask)
    39. # cv2.imshow("markerIni", markerIni)
    40. # cv2.imshow("imgRebuild", imgRebuild)
    41. # cv2.waitKey(0)
    42. # cv2.destroyAllWindows()
    43. # 统计重建后的图像中的连通域个数
    44. a = 1 # 腐蚀膨胀次数
    45. img = imgMask
    46. gray = img
    47. kernel = np.ones((2, 2), np.uint8) # 进行腐蚀膨胀操作 (2,2)的矩阵 1代表白色 0代表黑色 uint8代表8位无符号整数
    48. erosion = cv2.erode(gray, kernel, iterations=a) # 膨胀 iterations代表迭代次数
    49. dilation = cv2.dilate(erosion, kernel, iterations=a) # 腐蚀 先腐蚀再膨胀的目的是去除噪声
    50. ret, thresh = cv2.threshold(dilation, 150, 255, cv2.THRESH_BINARY) # 阈值处理 二值法 150代表阈值 255代表最大值 cv2.THRESH_BINARY代表二值法
    51. thresh1 = cv2.GaussianBlur(thresh, (15, 15), 0) # 高斯滤波 3,3代表高斯核的大小 0代表标准差
    52. contours, hirearchy = cv2.findContours(thresh1, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE) # 找出连通域
    53. # 对连通域面积进行比较
    54. area = [] # 建立空数组,放连通域面积
    55. contours1 = [] # 建立空数组,放减去最小面积的数
    56. for i in contours:
    57. # area.append(cv2.contourArea(i))
    58. # print(area)
    59. if cv2.contourArea(i) > 10: # 计算面积 去除面积小的 连通域
    60. contours1.append(i)
    61. # 输出细胞数目
    62. print(len(contours1))
    63. draw = cv2.drawContours(origin, contours1, -1, (0, 255, 0), 1) # 描绘连通域
    64. # 求连通域重心 以及 在重心坐标点描绘数字
    65. for i, j in zip(contours1, range(len(contours1))): # zip函数将两个数组合并
    66. M = cv2.moments(i) # 计算矩
    67. cX = int(M["m10"] / M["m00"]) # 计算重心横坐标
    68. cY = int(M["m01"] / M["m00"]) # 计算重心纵坐标
    69. draw1 = cv2.putText(draw, str(j), (cX, cY), 1, 1, (255, 0, 255), 1) # 在中心坐标点上描绘数字
    70. # 展示图片
    71. cv2.imshow("figure3:img_after_operation", thresh1)
    72. cv2.imshow("figure4:result", draw1)
    73. cv2.waitKey()

    Reference

    测地膨胀和腐蚀的概念:【图像处理笔记】形态学重建 - 湾仔码农 - 博客园

     例程 10.24:基于形态学重建的细胞计数 第二部分的代码来源于此:【youcans 的 OpenCV 例程200篇】134. 形态学重建之细胞计数_YouCans的博客-CSDN博客

     第二第三步的代码来源于此

    opencv——python细胞计数实现_microspore的博客-CSDN博客_opencv细胞计数

    附:失败尝试 

    如果不进行测地膨胀直接进行第二第三步会怎样

     

     效果不理想的代码:

    1. import cv2
    2. import numpy as np
    3. a=13
    4. img=cv2.imread(r'C:\Users\Administrator\PycharmProjects\pythonProject2\a.jpg',1) #读取图片
    5. #去除斑点
    6. gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) #将图片变为灰度图片
    7. kernel=np.ones((2,2),np.uint8) #进行腐蚀膨胀操作 (2,2)的矩阵 1代表白色 0代表黑色 uint8代表8位无符号整数
    8. erosion=cv2.erode(gray,kernel,iterations=a) #膨胀 iterations代表迭代次数
    9. dilation=cv2.dilate(erosion,kernel,iterations=a) #腐蚀
    10. ret, thresh = cv2.threshold(dilation, 150, 255, cv2.THRESH_BINARY) # 阈值处理 二值法 150代表阈值 255代表最大值 cv2.THRESH_BINARY代表二值法
    11. thresh1 = cv2.GaussianBlur(thresh,(3,3),0)# 高斯滤波 3,3代表高斯核的大小 0代表标准差
    12. contours,hirearchy=cv2.findContours(thresh1, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)# 找出连通域
    13. #对连通域面积进行比较
    14. area=[] #建立空数组,放连通域面积
    15. contours1=[] #建立空数组,放减去最小面积的数
    16. count=0 #计数
    17. total_area=0 #总面积
    18. for i in contours:
    19. total_area+=cv2.contourArea(i) #计算连通域面积
    20. avg_area=total_area/len(contours) #计算平均面积
    21. for i in contours:
    22. # area.append(cv2.contourArea(i))
    23. # print(area)
    24. if cv2.contourArea(i)>10: # 计算面积 去除面积小的 连通域
    25. contours1.append(i)
    26. count+=cv2.contourArea(i)/avg_area #计算面积比
    27. print(int(count)) #计算连通域个数
    28. draw=cv2.drawContours(img,contours1,-1,(0,255,0),1) #描绘连通域
    29. #求连通域重心 以及 在重心坐标点描绘数字
    30. for i,j in zip(contours1,range(len(contours1))): #zip函数将两个数组合并
    31. M = cv2.moments(i) # 计算矩
    32. cX=int(M["m10"]/M["m00"]) #计算重心横坐标
    33. cY=int(M["m01"]/M["m00"]) #计算重心纵坐标
    34. draw1=cv2.putText(draw, str(j), (cX, cY), 1,1, (255, 0, 255), 1) #在中心坐标点上描绘数字
    35. #展示图片
    36. cv2.imshow("draw",draw1)
    37. cv2.imshow("thresh1",thresh1)
    38. cv2.waitKey()
    39. cv2.destroyWindow()

  • 相关阅读:
    如何对服务器进行速度的测试
    C语言进阶第八课 --------通讯录的实现
    单页vue 列表图片懒加载 vue-lazyload 、html5
    LFS学习系列2 — 总览
    django+xadmin 在线教育网站(三)
    非父子组件通信-发布订阅模式
    武汉星起航:亚马逊卖家如何做好产品的差异化工作?
    Synchronized 与 Lock 生产者和消费者问题
    ⑱霍兰德ER*如何选专业?高考志愿填报选专业
    【2】c++11新特性(稳定性和兼容性)—>超长整型 long long
  • 原文地址:https://blog.csdn.net/qq_49860380/article/details/128032030