python canny检测_【数字图像分析】基于Python实现 Canny Edge Detection(Canny 边缘检测算法)...

Canny 边缘检测算法

Steps:

高斯滤波平滑

计算梯度大小和方向

非极大值抑制

双阈值检测和连接

代码结构:

Canny Edge Detection

|Gaussian_Smoothing

||convolution.py

|||convolution()

||gaussion_smoothing.py

|||dnorm()

|||gaussian_kernel()

|||gaussian_blur()

|Sobel_Filter

||sobel.py

|||sobel_edge_detection()

|Canny.py

||non_max_suppression()

||threshold()

||hysteresis()

||main()

代码解读:

1. 高斯滤波平滑

创建一个高斯核(kernel_size=5):

424b3182d69b297bf1d2e2858e7be3b8.png

执行卷积和平均操作(以下均以 lenna 图为例)

b9208a7953ba61c8197278004557a23f.png

2. 计算梯度大小和方向

水平方向和竖直方向

e647d858d3c058a2476709f73092620a.png

0776b59681d7c456be59ca16f9f4b7c7.png

梯度图:

e8f2f50c886217f53cec52a83c8331e5.png

3. 非极大值抑制

121d6b4fe7566e93ccddd3a5ffab7dc8.png

4. 双阈值检测和连接

b1b7f658f69a034d76fb468a15536ddf.png

以下是代码:

import numpy as np

import cv2

import argparse

from Computer_Vision.Canny_Edge_Detection.sobel import sobel_edge_detection

from Computer_Vision.Canny_Edge_Detection.gaussian_smoothing import gaussian_blur

import matplotlib.pyplot as plt

def non_max_suppression(gradient_magnitude, gradient_direction, verbose):

image_row, image_col = gradient_magnitude.shape

output = np.zeros(gradient_magnitude.shape)

PI = 180

for row in range(1, image_row - 1):

for col in range(1, image_col - 1):

direction = gradient_direction[row, col]

if (0 <= direction < PI / 8) or (15 * PI / 8 <= direction <= 2 * PI):

before_pixel = gradient_magnitude[row, col - 1]

after_pixel = gradient_magnitude[row, col + 1]

elif (PI / 8 <= direction < 3 * PI / 8) or (9 * PI / 8 <= direction < 11 * PI / 8):

before_pixel = gradient_magnitude[row + 1, col - 1]

after_pixel = gradient_magnitude[row - 1, col + 1]

elif (3 * PI / 8 <= direction < 5 * PI / 8) or (11 * PI / 8 <= direction < 13 * PI / 8):

before_pixel = gradient_magnitude[row - 1, col]

after_pixel = gradient_magnitude[row + 1, col]

else:

before_pixel = gradient_magnitude[row - 1, col - 1]

after_pixel = gradient_magnitude[row + 1, col + 1]

if gradient_magnitude[row, col] >= before_pixel and gradient_magnitude[row, col] >= after_pixel:

output[row, col] = gradient_magnitude[row, col]

if verbose:

plt.imshow(output, cmap='gray')

plt.title("Non Max Suppression")

plt.show()

return output

def threshold(image, low, high, weak, verbose=False):

output = np.zeros(image.shape)

strong = 255

strong_row, strong_col = np.where(image >= high)

weak_row, weak_col = np.where((image <= high) & (image >= low))

output[strong_row, strong_col] = strong

output[weak_row, weak_col] = weak

if verbose:

plt.imshow(output, cmap='gray')

plt.title("threshold")

plt.show()

return output

def hysteresis(image, weak):

image_row, image_col = image.shape

top_to_bottom = image.copy()

for row in range(1, image_row):

for col in range(1, image_col):

if top_to_bottom[row, col] == weak:

if top_to_bottom[row, col + 1] == 255 or top_to_bottom[row, col - 1] == 255 or top_to_bottom[row - 1, col] == 255 or top_to_bottom[

row + 1, col] == 255 or top_to_bottom[

row - 1, col - 1] == 255 or top_to_bottom[row + 1, col - 1] == 255 or top_to_bottom[row - 1, col + 1] == 255 or top_to_bottom[

row + 1, col + 1] == 255:

top_to_bottom[row, col] = 255

else:

top_to_bottom[row, col] = 0

bottom_to_top = image.copy()

for row in range(image_row - 1, 0, -1):

for col in range(image_col - 1, 0, -1):

if bottom_to_top[row, col] == weak:

if bottom_to_top[row, col + 1] == 255 or bottom_to_top[row, col - 1] == 255 or bottom_to_top[row - 1, col] == 255 or bottom_to_top[

row + 1, col] == 255 or bottom_to_top[

row - 1, col - 1] == 255 or bottom_to_top[row + 1, col - 1] == 255 or bottom_to_top[row - 1, col + 1] == 255 or bottom_to_top[

row + 1, col + 1] == 255:

bottom_to_top[row, col] = 255

else:

bottom_to_top[row, col] = 0

right_to_left = image.copy()

for row in range(1, image_row):

for col in range(image_col - 1, 0, -1):

if right_to_left[row, col] == weak:

if right_to_left[row, col + 1] == 255 or right_to_left[row, col - 1] == 255 or right_to_left[row - 1, col] == 255 or right_to_left[

row + 1, col] == 255 or right_to_left[

row - 1, col - 1] == 255 or right_to_left[row + 1, col - 1] == 255 or right_to_left[row - 1, col + 1] == 255 or right_to_left[

row + 1, col + 1] == 255:

right_to_left[row, col] = 255

else:

right_to_left[row, col] = 0

left_to_right = image.copy()

for row in range(image_row - 1, 0, -1):

for col in range(1, image_col):

if left_to_right[row, col] == weak:

if left_to_right[row, col + 1] == 255 or left_to_right[row, col - 1] == 255 or left_to_right[row - 1, col] == 255 or left_to_right[

row + 1, col] == 255 or left_to_right[

row - 1, col - 1] == 255 or left_to_right[row + 1, col - 1] == 255 or left_to_right[row - 1, col + 1] == 255 or left_to_right[

row + 1, col + 1] == 255:

left_to_right[row, col] = 255

else:

left_to_right[row, col] = 0

final_image = top_to_bottom + bottom_to_top + right_to_left + left_to_right

final_image[final_image > 255] = 255

return final_image

if __name__ == '__main__':

ap = argparse.ArgumentParser()

ap.add_argument("-i", "--image", required=True, help="Path to the image")

ap.add_argument("-v", "--verbose", type=bool, default=False, help="Path to the image")

args = vars(ap.parse_args())

image = cv2.imread(args["image"])

blurred_image = gaussian_blur(image, kernel_size=9, verbose=False)

edge_filter = np.array([[-1, 0, 1], [-2, 0, 2], [-1, 0, 1]])

gradient_magnitude, gradient_direction = sobel_edge_detection(blurred_image, edge_filter, convert_to_degree=True, verbose=args["verbose"])

new_image = non_max_suppression(gradient_magnitude, gradient_direction, verbose=args["verbose"])

weak = 50

new_image = threshold(new_image, 5, 20, weak=weak, verbose=args["verbose"])

new_image = hysteresis(new_image, weak)

plt.imshow(new_image, cmap='gray')

plt.title("Canny Edge Detector")

plt.show()

References

hahahha

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

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

相关文章

java 拖放文字_myeclipse2014如何实现jsp中的html代码的文字拖放

本帖最后由 liyihongcug 于 2015-3-5 16:53 编辑把 jsp打开 (visual jsp editor)之后上班区 单击右键 show --- pallette就可以了感叹myeclipse确实强大的 标签技术强于.net mvc强。vs2013能实现iis无需安装的情况下 动态指定 某个页面为首选 启动 --------------------希望2…

因缺思厅的绕过

看一下页面源码&#xff0c;看到source.txt。所以进入同目录下的source.txt 代码审计下&#xff0c;并且百度了一些函数。过滤了很多关键字&#xff0c;因此常规的SQL注入没有头绪。想了挺久&#xff0c;因为要满足三个条件。1&#xff1a;不能输入过滤的关键字2&#xff1a;只…

Python自制微信机器人:群发消息、自动接收好友

运营公众号也有半年了&#xff0c;今年5月份开始的&#xff0c;之前一直用一款windows工具来运营自动接受好友请求、群发文章、自动回复等操作&#xff0c;但颇有不便。 举几个场景&#xff1a; 突然在外面看到一篇文章很好&#xff0c;临时写了一篇&#xff0c;想群发一下。好…

记一次webpack4+react+antd项目优化打包文件体积的过程

背景 最近自己整了一个基于webpack4和react开发的博客demo项目&#xff0c;一路整下来磕磕碰碰但也实现了功能&#xff0c;就准备发到阿里云上面去看看&#xff0c;借用了同事的阿里云小水管服务器&#xff0c;配置完成之后首页加载花了十几秒&#xff0c;打开控制台network查看…

[19/03/16-星期六] 常用类_Date时间类DateFormat类

一、Date时间类 计算机中 以1970 年 1 月 1 日 00:00:00定为基准时间&#xff0c;每个度量单位是毫秒(1秒的千分之一) 用ong类型的变量来表示时间&#xff0c;如当前时刻数值&#xff1a;long now new System.currentTimeMillis(); 【常用方法】 1. Date() 分配一个Date对象&a…

实验一作业

Part1&#xff1a;实验总结体会 1.实验过程对程序编码的使用不熟练&#xff0c;经常在书写代码过程中有停顿与疑惑&#xff0c;但熟能生巧&#xff0c;经过训练还是有不少收获&#xff1b; 2.实验内容不够严谨&#xff0c;容易犯小错误&#xff0c;还是需要勤加练习&#xff1b…

php7引用计数,PHP7 引用计数 读书笔记

PHP7 引用计数 读书笔记每个 PHP 变量存在一个加 "zval" 的变量容器中. 一个 zval 变量容器, 除了包含变量的类型和值, 还包括两个字节的额外信息.1.is_ref : 是一个 bool 值, 用来标识这个变量是否属于引用集合. 通过这个字节, PHP 引擎才能把普通变量和引用变量区分…

焦呼?焦矣!

焦了。做了个香蕉蛋糕。总觉得它还没熟&#xff0c;就让它没完没了地烤着。于是焦了。可里面居然还是湿润的。是蛋糕的问题&#xff0c;还是6寸模的问题&#xff0c;还只是人的问题&#xff1f;烤一半时&#xff0c;爹临出门时&#xff0c;过来往上头按了一大拇指。很好&#x…

AJAX(2),我的第一个AJAX程序

首先&#xff0c;在VS2005中新建一个网站&#xff0c;拖入一个Button和Text,设置Button的客户端事件为userCheck(),<asp:TextBox ID"username"runat"server"></asp:TextBox><input type"button"name"check"value"…

[导入]ASP.NET 2.0中Page事件的执行顺序

文章来源:http://blog.csdn.net/21aspnet/archive/2007/03/20/1535517.aspx 转载于:https://www.cnblogs.com/zhaoxiaoyang2/archive/2007/03/21/816354.html

[导入]C#中TextBox只能输入数字的代码

文章来源:http://blog.csdn.net/21aspnet/archive/2007/03/20/1535640.aspx 转载于:https://www.cnblogs.com/zhaoxiaoyang2/archive/2007/03/21/816242.html

实验1c语言开发环境使用和数据类型、运算符和表达式

实验结论 由于这一次是第一次做实验有很多东西不熟悉 比如忘记加分号&#xff0c;用中文输入法打不对符号等等。总之经过这实验我学到了很多。#include <stdio.h> int main() {int x;printf("输入一个整数: \n");scanf("%d",&x);// 在处填写相应…

Nhibernate学习起步之many-to-one篇(转)

1. 学习目的: 通过进一步学习nhibernate基础知识&#xff0c;在实现单表CRUD的基础上&#xff0c;实现两表之间one-to-many的关系. 2. 开发环境必要准备 开发环境: windows 2003,Visual studio .Net 2005,Sql server 2005 developer edition 必要准备: 学习上篇文章单…

关于如何清除某个特定网站的缓存---基于Chrome浏览器

1、清除浏览器缓存 直接在浏览器设置里面清除浏览器的缓存会清除所有网站的缓存信息&#xff0c;这在某些时候是非常不方便的&#xff0c;毕竟不只有测试网站&#xff0c;还会有一些我们不想清除的信息也会被清除掉&#xff1b; 2、通过F12功能去清除浏览器缓存 转载于:https:/…

php中for循环流程图,PHP for循环

PHP for循环可以用来遍历一组指定的次数的代码。如果迭代次数已知&#xff0c;则应优先考虑使用for循环&#xff0c;否则使用while循环。for循环的语法for(initialization; condition; increment/decrement){ //code to be executed }for循环流程图示例代码-<?php for($n1;…

Spring中AOP切面编程学习笔记

注解方式实现aop我们主要分为如下几个步骤&#xff1a;  1.在切面类&#xff08;为切点服务的类&#xff09;前用Aspect注释修饰&#xff0c;声明为一个切面类。  2.用Pointcut注释声明一个切点&#xff0c;目的是为了告诉切面&#xff0c;谁是它的服务对象。&#xff08;此…

SpringCloud的学习记录(1)

最近一段时间重新学习一边SpringCloud&#xff0c;这里简单记录一下。 我用的是IntelliJ IDEA开发工具, SpringBoot的版本是2.1.3.RELEASE。 1. 构建Maven项目 整个的SpringCloud的项目是在Maven项目中的&#xff0c;这个Maven只做容纳其他项目使用, 比如后面Fegin/Config/Zipk…

html5 拖拽上传文件时,屏蔽浏览器默认打开文件

参考&#xff1a; https://www.cnblogs.com/kingsm/p/9849339.html 转载于:https://www.cnblogs.com/cwxwdm/p/10605529.html

php求数组交集的自定义函数,php数组交集函数

在数学中的交集运算&#xff0c;大家在学习的时候还是比较轻松的。我们在php数组里&#xff0c;可以借助array_intersect()函数对两个数组求交集&#xff0c;最后得到一个想要的交集数据。在正式开始array_intersect()使用前&#xff0c;我们需要先对函数的概念、语法、参数、返…

2016 hctf fheap 题解

题目链接 https://github.com/zh-explorer/hctf2016-fheap 题目分析 题目功能只有 malloc 和 free 的功能&#xff0c;查看堆的布局&#xff1a; 全局指针没有置空&#xff0c;导致 uaf 和 double free 漏洞&#xff0c;两种都可以做&#xff0c;但是这题的重点是绕过 PIE。 do…