当前位置: 首页 > news >正文

Python opencv识别图片中重叠圆的圆心位置

# -*- coding: utf-8 -*-
"""
只标注圆心 + 颜色一致性过滤 + 输出过程图思路:
1) 灰度 + Otsu 阈值;若圆为黑色前景,则反相使圆为白色前景(距离变换以白为前景)
2) 形态学闭运算,修补小孔/断裂
3) 欧氏距离变换(圆心处距离≈半径)
4) 距离模板相关得到峰值响应,阈值化为候选区域
5) 每个候选区域在距离图上取局部极大(圆心),并做“颜色一致性”过滤:- 圆心像素必须为前景(白)- 圆心附近小圆盘区域的前景占比 >= min_fg_ratio(默认 0.6)
6) 仅在圆心画十字;保存中间过程图
"""import os
import cv2
import numpy as np# ---------------- 配置参数 ----------------
img_path = 'code.jpg'
out_dir = 'out'                  # 输出过程图目录
invert_binary = True             # 若原图是白底黑圆,请保持 True(反相后圆=白色前景)
peak_thresh_ratio = 0.60         # 模板相关响应阈值(相对最大值比例)
min_peak_area = 5                # 峰值连通域最小面积(像素),过滤噪点
borderSize = 75                  # 模板/边界扩展参数
gap = 10                         # 模板与边缘的空隙
# 颜色一致性检查
disk_fraction = 0.5              # 用于颜色一致性检测的邻域半径 = r * disk_fraction(若 r 小则自适应 >= 4 像素)
min_fg_ratio = 0.60              # 邻域内白色前景比例阈值(0~1),越大越严格# 标注样式
marker_color = (0, 255, 0)
marker_size = 12
marker_thickness = 1def ensure_dir(d):if not os.path.exists(d):os.makedirs(d)def to_vis_u8(img_f32):"""将 float 图像线性归一化到 0~255 的 uint8,便于保存/显示"""v = cv2.normalize(img_f32, None, 0, 255, cv2.NORM_MINMAX)return v.astype(np.uint8)def fg_ratio_in_disk(fg_u8, cx, cy, r_eff):"""计算以 (cx, cy) 为圆心、半径 r_eff 的小圆盘内,前景(255)像素比例。fg_u8: 二值图或形态学结果,前景=255,背景=0"""h, w = fg_u8.shape[:2]if r_eff < 1:return 0.0x0, y0 = max(0, cx - r_eff), max(0, cy - r_eff)x1, y1 = min(w - 1, cx + r_eff), min(h - 1, cy + r_eff)roi = fg_u8[y0:y1+1, x0:x1+1]if roi.size == 0:return 0.0# 在 ROI 内生成圆形掩膜mask = np.zeros_like(roi, dtype=np.uint8)cv2.circle(mask, (cx - x0, cy - y0), r_eff, 255, thickness=-1)area = int(np.count_nonzero(mask))if area == 0:return 0.0# 前景像素计数(同时满足掩膜与前景)fg_count = int(np.count_nonzero(cv2.bitwise_and(roi, mask)))# 因为前景为 255,bitwise_and 后非零即计数;比例按像素个数计算return fg_count / areadef main():ensure_dir(out_dir)# 1) 读图与灰度im = cv2.imread(img_path, cv2.IMREAD_COLOR)if im is None:raise FileNotFoundError(f'无法读取图像:{img_path}')gray = cv2.cvtColor(im, cv2.COLOR_BGR2GRAY)# 2) 二值化(Otsu)_, bw = cv2.threshold(gray, 0, 255, cv2.THRESH_BINARY | cv2.THRESH_OTSU)# 反相:使圆为白色前景if invert_binary:bw = cv2.bitwise_not(bw)cv2.imwrite(os.path.join(out_dir, '01_bw.png'), bw)# 3) 形态学闭运算(修补小孔/细小断裂)kernel = cv2.getStructuringElement(cv2.MORPH_ELLIPSE, (3, 3))morph = cv2.morphologyEx(bw, cv2.MORPH_CLOSE, kernel)cv2.imwrite(os.path.join(out_dir, '02_morph.png'), morph)# 4) 距离变换dist = cv2.distanceTransform(morph, cv2.DIST_L2, 5)cv2.imwrite(os.path.join(out_dir, '03_dist.png'), to_vis_u8(dist))# 5) 构造模板并做归一化相关distborder = cv2.copyMakeBorder(dist, borderSize, borderSize, borderSize, borderSize,cv2.BORDER_CONSTANT | cv2.BORDER_ISOLATED, 0)kernel2 = cv2.getStructuringElement(cv2.MORPH_ELLIPSE, (2 * (borderSize - gap) + 1, 2 * (borderSize - gap) + 1))kernel2 = cv2.copyMakeBorder(kernel2, gap, gap, gap, gap,cv2.BORDER_CONSTANT | cv2.BORDER_ISOLATED, 0)distTempl = cv2.distanceTransform(kernel2, cv2.DIST_L2, 5)nxcor = cv2.matchTemplate(distborder, distTempl, cv2.TM_CCOEFF_NORMED)cv2.imwrite(os.path.join(out_dir, '04_nxcor.png'), to_vis_u8(nxcor))# 6) 阈值化得到峰值区域,并做最小面积过滤_, mx, _, _ = cv2.minMaxLoc(nxcor)_, peaks = cv2.threshold(nxcor, mx * peak_thresh_ratio, 255, cv2.THRESH_BINARY)peaks8u = cv2.convertScaleAbs(peaks)# 可选:删除太小的连通域(腐蚀/开运算或按轮廓面积过滤)cnt_res = cv2.findContours(peaks8u, cv2.RETR_CCOMP, cv2.CHAIN_APPROX_SIMPLE)contours = cnt_res[0] if len(cnt_res) == 2 else cnt_res[1]peaks_clean = np.zeros_like(peaks8u)kept_bboxes = []for cnt in contours:area = cv2.contourArea(cnt)if area >= min_peak_area:cv2.drawContours(peaks_clean, [cnt], -1, 255, thickness=-1)kept_bboxes.append(cv2.boundingRect(cnt))cv2.imwrite(os.path.join(out_dir, '05_peaks_mask.png'), peaks_clean)# 7) 在每个候选区域中找局部极大,并做颜色一致性过滤annotated = im.copy()centers = []for index, (x, y, w_box, h_box) in enumerate(kept_bboxes):# 用峰值掩膜作为 mask,限定在候选区域内找最大值(圆心)mask_roi = peaks_clean[y:y+h_box, x:x+w_box]min_val, max_val, min_loc, max_loc = cv2.minMaxLoc(dist[y:y+h_box, x:x+w_box], mask=mask_roi)cx, cy = int(max_loc[0] + x), int(max_loc[1] + y)r = int(round(max_val))  # 估计半径# --- 颜色一致性过滤 ---# 1) 圆心像素必须为前景(白)if morph[cy, cx] == 0:continue# 2) 圆心邻域的前景占比必须足够高r_eff = max(4, int(r * disk_fraction))fg_ratio = fg_ratio_in_disk(morph, cx, cy, r_eff)if fg_ratio < min_fg_ratio:continuecenters.append((cx, cy))# 只画圆心(绿色十字)cv2.drawMarker(annotated, (cx, cy),color=marker_color,markerType=cv2.MARKER_CROSS,markerSize=marker_size,thickness=marker_thickness,line_type=cv2.LINE_AA)cv2.putText(annotated, f'#{index+1} ({cx}, {cy})', (cx + 5, cy - 5),cv2.FONT_HERSHEY_SIMPLEX, 0.5, marker_color, 1, cv2.LINE_AA)# 8) 输出print(f'最终圆心数:{len(centers)}')for i, (cx, cy) in enumerate(centers, start=1):print(f'#{i}: x={cx}, y={cy}')cv2.imwrite(os.path.join(out_dir, '06_centers.png'), annotated)cv2.imshow('centers', annotated)cv2.waitKey(0)cv2.destroyAllWindows()if __name__ == '__main__':main()

在这里插入图片描述

在这里插入图片描述

http://www.dtcms.com/a/343736.html

相关文章:

  • 焊接工艺仿真软件SYSWELD、Simufact.Welding和FLOW-3D WELD的区别
  • 【0基础3ds Max】捕捉工具详解
  • 城市生命线地下管网三维建模软件MagicPipe3D更新历史
  • ZBrush和3D-Coat如何实现快速桥接?
  • `git mv` 重命名 Git 仓库中的文件夹
  • OFSP路由与传输层协议
  • LLM实践系列:利用LLM重构数据科学流程
  • 亚矩阵云手机:亚马逊第三方店铺多账号安全合规运营的核心技术支撑
  • 代码随想录Day59:图论(最短路算法dijkstra堆优化版精讲、Bellman_ford 算法精讲)
  • 【生成树+环】题解:P3907 环的异或_图论_环_异或_搜索_算法竞赛_C++
  • 【typenum】 22 类型级别二进制对数运算(Logarithm2)
  • Apache Ozone 介绍与部署使用(最新版2.0.0)
  • Vue2+Vue3前端开发_Day6
  • Spring-AI初级使用记录 spring-ai-bom版本1.0.1-(单、多轮对话)
  • reactive 核心要点
  • FFmpeg及 RTSP、RTMP
  • 大型前端项目如何实现css 隔离:利用浏览器原生的 Shadow DOM 完全隔离 DOM 结构与样式...
  • 前端AI工具——TRAE
  • Linux基础命令大全:从入门到熟练
  • 开发避坑指南(34):mysql深度分页查询优化方案
  • GitCode 疑难问题诊疗:全面指南与解决方案
  • 关于在 IntelliJ IDEA 中安装和配置 Java 17
  • 简单聊聊多模态大语言模型MLLM
  • RabbitMQ 应用问题
  • RabbitMQ深度剖析:从基础到高级进阶实战
  • RabbitMQ 全面指南:架构解析与案例实战
  • 线性回归学习笔记
  • k8s——持久化存储 PVC
  • 自定义rabbitmq的ConnectionFactory配置
  • uniapp轮播 轮播图内有定位样式