图片中线段和圆圈检测(python opencv)

在这里插入图片描述

前言

本文实现将一个图片中的线段和圆圈检测出来,效果就像这样

企业微信截图_17046860711812.png

开始之前请先自行安装 opencv

另外还用到了一个用来检测直线: http://olena.pages.lre.epita.fr/pylena/index.html

pip install pylena

直线检测

先用 opencv 来检测直线, 因为下面代码是运行在 google 实验室的,所以显示图片使用 google.colab.patches 提供的方法

import cv2
import numpy as np
from google.colab.patches import cv2_imshow
import sys
import math
import matplotlib.pyplot as plt

# 读取原始图像 
image = cv2.imread("/content/2.jpg") 

# 将图像转换为灰度
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)

kernel = np.ones((1))
img_dilated = cv2.dilate(gray, kernel, iterations=1) 

# 使用Canny边缘检测算法 
edges = cv2.Canny(img_dilated, 100, 200)

# 使用Hough线变换算法检测线
lines = cv2.HoughLinesP(edges,1,np.pi/180, threshold=25, minLineLength=20, maxLineGap=20)

# 绘制检测到的线
for line in lines:
    x1, y1, x2, y2 = line[0]
    cv2.line(image, (x1, y1), (x2, y2), (0, 0, 255), 2)
 
# 显示结果
print(f"共: {len(lines)} 条线段")
cv2_imshow(image)
 

效果是这样的,虽然也都差不太多了,但是还得手动处理一些数据才行,因为重复的和没用的线段有些多,都需要删除掉才行。
企业微信截图_17046763521427.png

那就换一个库,使用 pylena

import matplotlib.pyplot as plt
from skimage.data import text
from skimage import io, color
import pylena as pln
import numpy as np 
from skimage import img_as_ubyte  # 用于将图像转换为 uint8 类型
from typing import List
import random
import cv2 


def random_red_color():
    # 随机生成红色通道值
    red_channel = 255
    green_channel = random.randint(0, 255)
    blue_channel = random.randint(0, 255)

    # 返回RGB颜色
    return (red_channel, green_channel, blue_channel)

# 转换为 uint8 类型
img2 = img_as_ubyte(color.rgb2gray(img.copy())) 
plt.xticks([])
plt.yticks([]) 

img_label: np.ndarray
superpositions: List[pln.scribo.LSuperposition]
lines: List[pln.scribo.VSegment]
 
img_label,superpositions, lines = pln.scribo.line_detector(
    img2, 
    "full", verbose=False, 
    min_len=50, 
    blumi=160, 
    llumi=160, 
    discontinuity_relative=0, 
    minimum_for_fusion=0
)
print(f"共检测出:{len(lines)}条线段")
 
if lines is not None:
    for line in lines:  
        cv2.line(img, (line.x0, line.y0), (line.x1, line.y1), random_red_color(), 2) 
else:
    print("未检测出直线")


plt.xticks([])
plt.yticks([]) 

plt.imshow(img)
plt.show()
 

效果如下,但是每一次检测都是不一样的结果,差距倒是不大。
image.png

完善下代码,将圆圈也检测一些,圆圈检测使用 opencv

import matplotlib.pyplot as plt
from skimage.data import text
from skimage import io, color
import pylena as pln
import numpy as np 
from skimage import img_as_ubyte  # 用于将图像转换为 uint8 类型
from typing import List
import random
import cv2


def random_red_color():
    # 随机生成红色通道值
    red_channel = 255
    green_channel = random.randint(0, 255)
    blue_channel = random.randint(0, 255)

    # 返回RGB颜色
    return (red_channel, green_channel, blue_channel)


# ==== 圆形检测 ==== 
img = io.imread("/content/1.jpg")
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
circles = cv2.HoughCircles(
    gray,
    cv2.HOUGH_GRADIENT,
    dp=1,
    minDist=50,
    param1=100,
    param2=25,
    minRadius=5,
    maxRadius=100
)
print(f"共检测出:{len(circles[0])}个圆") 
if circles is not None:
    circles = np.uint16(np.around(circles))
    for circle in circles[0, :]:
        center = (circle[0], circle[1])
        # radius = circle[2]
        radius = 30 # 别超过下面直线的最大检测长度
        cv2.circle(img, center, radius, random_red_color(), 2)
else:
    print("未检测出圆圈")

# plt.imshow(img)
# plt.show()



# ==== 直线检测 ==== 
# 转换为 uint8 类型
img2 = img_as_ubyte(color.rgb2gray(img.copy())) 
plt.xticks([])
plt.yticks([]) 

img_label: np.ndarray
superpositions: List[pln.scribo.LSuperposition]
lines: List[pln.scribo.VSegment]
 
img_label,superpositions, lines = pln.scribo.line_detector(
    img2, 
    "full", verbose=False, 
    min_len=50, 
    blumi=160, 
    llumi=160, 
    discontinuity_relative=0, 
    minimum_for_fusion=0
)
print(f"共检测出:{len(lines)}条线段")
 
if lines is not None:
    for line in lines:  
        cv2.line(img, (line.x0, line.y0), (line.x1, line.y1), random_red_color(), 2) 
else:
    print("未检测出直线")
 
plt.imshow(img)
plt.show()


image.png

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

天融信 2023 的年终奖。。

天融信 过去几天,最大的瓜,是天融信 2023 的年终奖脚踝砍。 "天融信"是国内首家网络安全企业,同时也是一家上市公司。 就在前些天,有网友爆料出,天融信年终奖到账只有几百元,甚至只有几十元&…

铝包木门窗性能优异 国内产量及需求量总体呈增长态势

铝包木门窗性能优异 国内产量及需求量总体呈增长态势 铝包木门窗是在保留纯实木门窗特性和功能的前提下,将隔热断桥铝合金型材和实木通过机械方法复合而成的框体。铝包木门窗具有良好的密封性、保温性、抗腐蚀性、隔音性等,能够满足市场对门窗质量要求不…

【Linux-阻塞,非阻塞,异步】

Linux-阻塞,非阻塞,异步 ■ Linux-阻塞-非阻塞 IO-异步■ Linux-阻塞IO■ 阻塞IO简介■ open■ 等待队列■ 示例一:APP阻塞IO读取进入睡眠后-等待队列唤醒进程■■ ■ Linux-非阻塞IO■ 非阻塞IO简介■ open■ 轮询■ 1、select 函数■ 2、po…

【轻触按键】开关机电路--填坑1

接上文,挖的坑 一、翻转电路 二、真值表 按键按下去,1G17会拉低,A端脚会掉电,下降沿;终到逻辑“0” 松开按键,1G17会拉高,A端脚充电,上升沿;终到逻辑“1”; …

[补题记录]LeetCode 6.Z字形变换

传送门:Z字形变换 转自:Z字形变换 Thought/思路 关键点在于,最后的答案是一行行连接起来的。 这样我们就会发现,这个 Z 字,实际上会让行数 不断加 1,然后又 不断减 1。每次按顺序选择 S 中的一个字符即…

visual studio打包qt算子时,只生成dll没有生成lib等文件

问题:在visual studio配置了qt项目,并打包成dll,原则上会生成一堆文件,包括dll,lib等文件。 解决办法: 挨个右击源代码的所有头文件-》属性-》项类型。改成qt头文件形式,如下。

961题库 北航计算机 计算机网络 附答案 简答题形式

有题目和答案,没有解析,不懂的题问大模型即可,无偿分享。 第1组 习题 某网络拓扑如题下图所示,其中 R 为路由器,主机 H1~H4 的 IP 地址配置以及 R 的各接口 IP 地址配置如图中所示。现有若干以太网交换机…

C#中使用Mapster

Mapster是一个开源的.NET对象映射库,它提供了一种简单而强大的方式来处理对象之间的映射。 多个映射框架的性能对比: 第一步安装Mapster 使用方法 public class Test {public string name { get; set; }public string sex { get; set; }public string…

光伏并网逆变器UL 1741:2021标准解析

光伏并网逆变器UL 1741:2021标准解析 不同国家的安规认证可以说是光伏逆变器走向国际市场的一张通行证,由于全球各国家的电网制式及并网政策的不同差异,这对逆变器测试顺利的通过安规测试认证 还是有一定的技术难度,也是中国光伏制造企业迫切…

在Linux中tomcat占用内存过高可以通过导出hprof日志来解决

自动导出hprof日志 第一种方法: Tomcat的hprof日志是一种用于分析Java堆内存使用情况的工具,它可以帮助开发人员找到内存泄漏的原因。 hprof日志可以在特定的时间点对Java堆内存进行快照,并生成详细的分析报告。 启用hprof日志导出的具体…

零基础写框架:从零设计一个模块化和自动服务注册框架

模块化和自动服务注册 基于 ASP.NET Core 开发的 Web 框架中,最著名的是 ABP,ABP 主要特点之一开发不同项目(程序集)时,在每个项目中创建一个模块类,程序加载每个程序集中,扫描出所有的模块类,然后通过模块…

“去员工化”这个潮流谁也挡不住,六大诱因分析。

去员工化→恐怕是未来工作的主流,一方面有成本的原因,另一方面也有技术进步、雇佣形式创新等原因,这个潮流有利也有弊,关键看我们是如何兴利除弊。 "去员工化"是指企业在运营中减少或取消传统雇佣制度,更多…

HOW - vscode 使用指南

目录 一、基本介绍1. 安装 VS Code2. 界面介绍3. 扩展和插件4. 设置和自定义 二、常用界面功能和快捷操作(重点)常用界面功能快捷操作 三、资源和支持 Visual Studio Code(VS Code)是一款由微软开发的免费、开源的代码编辑器&…

vue开发网站-使用插件element、vant 遇到的问题

1. js把两个字符串放进一个另字符串里,用逗号分隔 let string1 "Hello"; let string2 "World"; let result ${string1},${string2}; console.log(result); // 输出: Hello,World2.js将字符串转为数组 const str "Hello, world!"…

HBuilder中怎样修改每次输出的内容hello?每次修改之后再次“运行到”的时候,怎样保证每次都重新编译?

要修改每次输出的内容,可以在代码中找到输出的地方,并将内容进行修改。例如,在JavaScript中,可以使用console.log()函数输出内容。在修改内容后,保存代码。 为了保证每次运行都重新编译代码,可以按照以下步…

LeetCode 算法:接雨水c++

原题链接🔗:接雨水 难度:困难⭐️⭐️⭐️ 题目 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 示例 1: 输入:height [0,1,0,2,1,0,1,3,2,…

使用 WM_WINDOWPOSCHANGING 跟踪窗口状态变化

在窗口位置变化过程的早期,系统会发送 WM_WINDOWPOSCHANGING 消息。 这个和 WM_WINDOWPOSCHANGED 消息不同,WM_WINDOWPOSCHANGED 消息发生在窗口位置变化之后。 一个关键的区别(除了时间之外)是,您可以通过处理 WM_WI…

OpenStreetMap部署(OSM)

参考:https://github.com/openstreetmap/openstreetmap-website/blob/master/DOCKER.md OpenStreeMap 部署 操作系统建议使用 Ubuntu 22 版本 安装 Docker # 更新软件包索引: sudo apt-get update # 允许APT使用HTTPS: sudo apt-get inst…

艰难求生的转型之路

起因 我个人“工作水平低,专业能力差。”是最核心的困难。 在坚持了快十年之后,博客从2015-2024。 2015201620172018201920202021202220232024 从2020年之后就已经开始全面转型之路。 所有传统赛道,都挤满了人,各种限制各种约…

【图像处理与机器视觉】灰度变化与空间滤波

基础 空间域与变换域 空间域:认为是图像本身,对于空间域的操作就是对图像中的像素直接进行修改 变换域:变换系数处理,不直接对于图像的像素进行处理 邻域 图像中某点的邻域被认为是包含该点的小区域,也被称为窗口 …