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

news2025/8/9 23:38:03

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

step1:先进行测地膨胀

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

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

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

图2:测地膨胀后的结果

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

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

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

图4:最终结果

 完整代码

import cv2
import numpy as np
import matplotlib.pyplot as plt

origin = cv2.imread("a.jpg")
imgGray = cv2.imread("origin", flags=0)  # flags=0 灰度图像
ret, imgBin = cv2.threshold(imgGray, 205, 255, cv2.THRESH_BINARY)  # 二值化处理 (黑色0/白色1) ret:阈值 205:阈值 255:最大值 cv2.THRESH_BINARY:二值化
imgMask = cv2.bitwise_not(imgBin)  # 二值图像的补集 (黑色1/白色0)

# 构造标记图像
marker = np.zeros_like(imgBin, dtype=np.uint8) # 与imgBin同尺寸的全0图像
marker[0, :] = imgBin[0, :]  # 第一行
marker[-1, :] = imgBin[-1, :]  # 最后一行
marker[:, 0] = imgBin[:, 0]  # 第一列
marker[:, -1] = imgBin[:, -1]  # 最后一列
markerIni = marker.copy()  # 标记图像: 边框 f(x,y)=I(x,y),其它为 0

# 形态学重建
element = cv2.getStructuringElement(cv2.MORPH_CROSS, (3, 3))
#创建数组画出迭代图
diffList = []
while True: #进行测地膨胀 是有条件的膨胀
    marker_pre = marker  # 保存 F(n-1)
    dilation = cv2.dilate(marker, kernel=element)  # 膨胀重建 目的是扩大 F(n-1) 的边界
    marker = cv2.bitwise_and(dilation, imgMask)  # 原图像作为模板用来约束重建,按位与,有 0 得 0 目的是保持 F(n-1) 的内部
    #计算重建后的标记图像与重建前的标记图像的差异
    diff = cv2.subtract(marker, marker_pre)
    print(diff.sum())
    diffList.append(diff.sum())
    if (marker_pre == marker).all():  # F(n)=F(n-1)?,判断是否达到稳定收敛状态
        break
imgRebuild = cv2.bitwise_or(imgBin, marker)  # 按位或,有 1 得 1
#绘制迭代图,命名为"图1:多次迭代的图像差异"
plt.figure("图1:多次迭代的图像差异")
plt.plot(diffList)
plt.show()


# 显示
# cv2.imshow("imgGray", imgGray)
# cv2.imshow("imgBin", imgBin)
cv2.imshow("figure2:after Expansion", imgMask)
# cv2.imshow("markerIni", markerIni)
# cv2.imshow("imgRebuild", imgRebuild)
# cv2.waitKey(0)
# cv2.destroyAllWindows()


#统计重建后的图像中的连通域个数
a=1 #腐蚀膨胀次数
img=imgMask
gray=img
kernel=np.ones((2,2),np.uint8) #进行腐蚀膨胀操作 (2,2)的矩阵 1代表白色 0代表黑色 uint8代表8位无符号整数
erosion=cv2.erode(gray,kernel,iterations=a) #膨胀 iterations代表迭代次数
dilation=cv2.dilate(erosion,kernel,iterations=a) #腐蚀 先腐蚀再膨胀的目的是去除噪声
ret, thresh = cv2.threshold(dilation, 150, 255, cv2.THRESH_BINARY) # 阈值处理 二值法 150代表阈值 255代表最大值 cv2.THRESH_BINARY代表二值法
thresh1 = cv2.GaussianBlur(thresh,(15,15),0) # 高斯滤波 3,3代表高斯核的大小 0代表标准差
contours,hirearchy=cv2.findContours(thresh1, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)# 找出连通域
#对连通域面积进行比较
area=[] #建立空数组,放连通域面积
contours1=[]   #建立空数组,放减去最小面积的数

for i in contours:
      # area.append(cv2.contourArea(i))
      # print(area)
     if cv2.contourArea(i)>10:  # 计算面积 去除面积小的 连通域
        contours1.append(i)
#输出细胞数目
print(len(contours1))
draw=cv2.drawContours(origin,contours1,-1,(0,255,0),1) #描绘连通域
#求连通域重心 以及 在重心坐标点描绘数字
for i,j in zip(contours1,range(len(contours1))): #zip函数将两个数组合并
    M = cv2.moments(i) # 计算矩
    cX=int(M["m10"]/M["m00"]) #计算重心横坐标
    cY=int(M["m01"]/M["m00"]) #计算重心纵坐标
    draw1=cv2.putText(draw, str(j), (cX, cY), 1,1, (255, 0, 255), 1) #在中心坐标点上描绘数字
#展示图片
cv2.imshow("figure3:img_after_operation",thresh1)
cv2.imshow("figure4:result",draw1)
cv2.waitKey()

Reference

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

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

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

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

附:失败尝试 

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

 

 效果不理想的代码:

import cv2
import numpy as  np
a=13
img=cv2.imread(r'C:\Users\Administrator\PycharmProjects\pythonProject2\a.jpg',1) #读取图片
#去除斑点

gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) #将图片变为灰度图片
kernel=np.ones((2,2),np.uint8) #进行腐蚀膨胀操作 (2,2)的矩阵 1代表白色 0代表黑色 uint8代表8位无符号整数
erosion=cv2.erode(gray,kernel,iterations=a) #膨胀 iterations代表迭代次数
dilation=cv2.dilate(erosion,kernel,iterations=a) #腐蚀
ret, thresh = cv2.threshold(dilation, 150, 255, cv2.THRESH_BINARY) # 阈值处理 二值法 150代表阈值 255代表最大值 cv2.THRESH_BINARY代表二值法
thresh1 = cv2.GaussianBlur(thresh,(3,3),0)# 高斯滤波 3,3代表高斯核的大小 0代表标准差
contours,hirearchy=cv2.findContours(thresh1, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)# 找出连通域
#对连通域面积进行比较
area=[] #建立空数组,放连通域面积
contours1=[]   #建立空数组,放减去最小面积的数
count=0 #计数
total_area=0 #总面积
for i in contours:
    total_area+=cv2.contourArea(i) #计算连通域面积
avg_area=total_area/len(contours) #计算平均面积
for i in contours:
      # area.append(cv2.contourArea(i))
      # print(area)
     if cv2.contourArea(i)>10:  # 计算面积 去除面积小的 连通域
        contours1.append(i)
        count+=cv2.contourArea(i)/avg_area #计算面积比
print(int(count)) #计算连通域个数
draw=cv2.drawContours(img,contours1,-1,(0,255,0),1) #描绘连通域
#求连通域重心 以及 在重心坐标点描绘数字
for i,j in zip(contours1,range(len(contours1))): #zip函数将两个数组合并
    M = cv2.moments(i) # 计算矩
    cX=int(M["m10"]/M["m00"]) #计算重心横坐标
    cY=int(M["m01"]/M["m00"]) #计算重心纵坐标
    draw1=cv2.putText(draw, str(j), (cX, cY), 1,1, (255, 0, 255), 1) #在中心坐标点上描绘数字
#展示图片
cv2.imshow("draw",draw1)
cv2.imshow("thresh1",thresh1)
cv2.waitKey()
cv2.destroyWindow()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/34007.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ssm分页实战

1. 插件 1. maven <!-- 分页插件 --><dependency><groupId>com.github.pagehelper</groupId><artifactId>pagehelper</artifactId><version>5.2.0</version></dependency> 3. mybaits核心配置文件中或spring配置文件中…

这就叫速度,并发编程深度解析实战七天杀上 GitHub 榜首

并发编程&#xff01;在一二线互联网公司的面试中&#xff0c;逃避不了也是必问的面试题&#xff0c;而绝大部分程序员对并发编程的理解也都停留在使用阶段。那么今天团长给大家推荐的这份资料&#xff0c;绝对可以帮到你。 本书涵盖了六个特点&#xff1a; 特色一&#xff1a…

linux mysql5.7.25 主从复制_生产版本

文章目录一、安装配置1. 部署总览2. 下载软件3. 解压重命名4. 创建组5. 安装数据库6. 配置my.cnf7. 添加开机启动8. 配置数据木库9. 启动mysql10. 登录修改密码11. 允许远程连接二、master节点主从2.1. 主从复制账号创建2.2. 创建主从复制的账号2.3. 账号授权2.4. 查看同步bin-…

VUE的10个常用指令

01 v-once 说明&#xff1a;只渲染元素和组件一次。随后的重新渲染&#xff0c;元素/组件及其所有的子节点将被视为静态内容并跳过。这可以用于优化更新性能。 <!-- 单个元素 --> <span v-once>This will never change: {{msg}}</span> <!-- 有子元素 --&…

二进制逻辑运算和基本门电路

目录 基本门电路很重要&#xff0c;做内存扩展片选译码的时候会常用 一&#xff1a;逻辑非&#xff08;按位取反&#xff09; not 二&#xff1a;逻辑乘&#xff08;逻辑与&#xff09;按位求“与” 有零出零 and 三&#xff1a;逻辑或&#xff08;逻辑加) 有1出1 …

2023-2028年中国化工新材料行业发展前景与投资趋势分析报告

本报告由锐观咨询重磅推出&#xff0c;对中国化工新材料行业的发展现状、竞争格局及市场供需形势进行了具体分析&#xff0c;并从行业的政策环境、经济环境、社会环境及技术环境等方面分析行业面临的机遇及挑战。还重点分析了重点企业的经营现状及发展格局&#xff0c;并对未来…

JavaScript高级复习下(60th)

1、函数内 this 的指向 2、严格模式 1、什么是严格模式 JavaScript 除了提供正常模式外&#xff0c;还提供了 严格模式&#xff08;strict mode&#xff09;。ES5 的严格模式是采用具有限制性 JavaScript 变体的一种方式&#xff0c;即在严格的条件下运行 JS 代码。 严格模式…

说一下 ArrayDeque 和 LinkedList 的区别?

大家好&#xff0c;我是小彭。 在上一篇文章里&#xff0c;我们聊到了基于链表的 Queue 和 Stack 实现 —— LinkedList。那么 Java 中有没有基于数组的 Queue 和 Stack 实现呢&#xff1f;今天我们就来聊聊这个话题。 小彭的 Android 交流群 02 群已经建立啦&#xff0c;扫描…

APP到底有没有权限-恶意拷贝删除照片

作者&#xff1a;黑蛋 近期发生了一件比较恶劣的事情&#xff0c;某客户在某物上面买了一件东西&#xff0c;但是这个东西是假货&#xff0c;所以客户致电某物人工客服&#xff0c;并进行了录音&#xff0c;这时候某物试图通过自身的客户端软件&#xff0c;去删除客户手机上的…

DFP 数据转发协议应用实例 7.使用 DLS1x 与 VSxxx 设备的 LoRA 匹配

DFP 数据转发协议应用实例 7.使用 DLS1x 与 VSxxx 设备的 LoRA 匹配 DFP 是什么&#xff1f; 稳控科技编写的一套数据转发规则&#xff0c; 取自“自由转发协议 FFP&#xff08;Free Forward Protocol&#xff09;” &#xff0c;或者 DFP&#xff08;DoubleF Protocol&#x…

维格云单点登录SSO入门教程

功能简介 无代码维格云单点登录功能支持用户通过配置,将无代码维格云的帐号体系,和提供标准OAuth2.0认证服务系统、LDAP的帐号体系统一起来。 两种单点登录方式的配置见子文档:OAuth 2.0LDAP,本文只描述通用功能:全局单点登录、退出登录后跳转指定页面。 注:本功能为A…

ArcGIS综合制图教程,简单上手!

目的 1、了解专题地图组成的各个要素&#xff1b; 2、掌握ArcGIS编辑专题图的方法和步骤&#xff1b; 实习内容 使用ArcGIS生成1&#xff1a;200万比例尺的浙江省县级行政区划图&#xff0c;并输出成像文件。 实习步骤 一、将ArcGIS切换到Layout视图&#xff0c;并调整页面…

HTML、CSS学习笔记小结

目录 1&#xff0c;HTML 1.1 简单介绍 1.2 快速入门 1.3 基础标签 代码演示&#xff1a; 1.4 图片、音频、视频标签 代码演示&#xff1a; 1.5 超链接标签 代码演示&#xff1a; 1.6 列表标签 代码演示&#xff1a; 1.7 表格标签 代码演示&#xff1a; 1.8 布局标…

【LeetCode与《代码随想录》】哈希表篇:做题笔记与总结-JavaScript版

文章目录代码随想录主要题目242. 有效的字母异位词349. 两个数组的交集202. 快乐数1. 两数之和&#xff08;经典哈希&#xff09;454. 四数相加 II15. 三数之和&#xff08;双指针&#xff09;18. 四数之和&#xff08;双指针&#xff09;相关题目383. 赎金信49. 字母异位词分组…

【构建ML驱动的应用程序】第 9 章 :选择部署选项

&#x1f50e;大家好&#xff0c;我是Sonhhxg_柒&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流&#x1f50e; &#x1f4dd;个人主页&#xff0d;Sonhhxg_柒的博客_CSDN博客 &#x1f4c3; &#x1f381;欢迎各位→点赞…

Redis基础命令(set类型)交集并集差集

目录 概述: 特征&#xff1a; Set常见命令&#xff1a; 1.Sadd key number..&#xff1a;向set中添加一个或多个元素 2.Srem key number...&#xff1a;移除set中指定的元素 3.Scard key&#xff1a;返回set中元素的个数 4.Sismember key member&#xff1a;判断一个元素…

liunx中如何启动redis

连上服务器之后&#xff0c;这个“~”’波浪符号应该是在root目录下&#xff0c;我们需要到这个目录的外面&#xff0c;根目录 root目录 目录的外面&#xff0c;根目录 cd .. 变为"/"符号即可 然后再找到redis的安装目录&#xff0c;也可以在xftp中找&#xff0c;可视…

【LeetCode每日一题】——90.子集 II

文章目录一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【解题思路】七【题目提示】八【时间频度】九【代码实现】十【提交结果】一【题目类别】 数组 二【题目难度】 中等 三【题目编号】 90.子集 II 四【题目描述】 给你一个整数数组 num…

有没有把语音转为文字的软件?这几个转换软件你值得收藏

我们在日常的工作和生活中&#xff0c;应该经常会遇到需要将音频转换成文字的情况吧。相信大部分的小伙伴都会选择直接使用转换软件进行音频转文字的操作&#xff0c;但在使用的过程中就会发现&#xff0c;有些软件会在使用次数、音频时长上面有所限制&#xff0c;导致我们会转…

【文件格式_XML_HTML_】XML、HTML文件

HTML xml what&#xff1f; XML是被设计用来存储数据、携带数据和交换数据的&#xff0c;通过XML&#xff0c;可以在不兼容的系统之间交换数据&#xff0c;利用XML&#xff0c;纯文本文件可以用来存储数据。在不使用XML时&#xff0c;HTML用于显示数据&#xff0c;数据必须存…