如何实现视觉识别形状

news2024/12/23 12:01:57

1. 功能说明

通过摄像头识别圆形及矩形两种形状。

 2. 电子硬件

     本实验中采用了以下硬件:

主控板

Basra主控板(兼容Arduino Uno)

扩展板

Bigfish2.1

电池7.4V锂电池
通信2510通信转接板
WiFi路由器

其它

摄像头

配置OpenCV的Visual Studio 2015.net环境的计算机一台

3. 功能实现  

   工作原理:

                  ① 导入一张图片或者通过WiFi传递摄像信息给PC;

                  ② 在PC端使用OpenCV对图像转化为灰度图像;

                  ③ 检测圆形和矩形。

    检测圆形使用霍夫变换:

vector<Vec3f> circles;

HoughCircles(image, circles, HOUGH_GRADIENT, 2.0,

image.rows/8, // change this value to detect circles with different distances to each other

200, 85, 0, 0 // change the last two parameters

  // (min_radius & max_radius) to detect larger circles

);

    检测矩形:

//多边形检测,通过约束条件寻找矩形

static void findSquares(const Mat& image, vector<vector<Point> >& squares)

{

squares.clear();


Mat pyr, timg, gray0(image.size(), CV_8U), gray;


// down-scale and upscale the image to filter out the noise

pyrDown(image, pyr, Size(image.cols / 2, image.rows / 2));

pyrUp(pyr, timg, image.size());

vector<vector<Point> > contours;


// find squares in every color plane of the image

for (int c = 0; c < 3; c++)

{

int ch[] = { c, 0 };

mixChannels(&timg, 1, &gray0, 1, ch, 1);


// try several threshold levels

for (int l = 0; l < N; l++)

{

// hack: use Canny instead of zero threshold level.

// Canny helps to catch squares with gradient shading

if (l == 0)

{

// apply Canny. Take the upper threshold from slider

// and set the lower to 0 (which forces edges merging)

Canny(gray0, gray, 0, thresh, 5);

// dilate canny output to remove potential

// holes between edge segments

dilate(gray, gray, Mat(), Point(-1, -1));

}

else

{

// apply threshold if l!=0:

//     tgray(x,y) = gray(x,y) < (l+1)*255/N ? 255 : 0

gray = gray0 >= (l + 1) * 255 / N;

}


// find contours and store them all as a list

findContours(gray, contours, RETR_LIST, CHAIN_APPROX_SIMPLE);


vector<Point> approx;


// test each contour

for (size_t i = 0; i < contours.size(); i++)

{

// approximate contour with accuracy proportional

// to the contour perimeter

approxPolyDP(Mat(contours[i]), approx, arcLength(Mat(contours[i]), true)*0.02, true);


// square contours should have 4 vertices after approximation

// relatively large area (to filter out noisy contours)

// and be convex.

// Note: absolute value of an area is used because

// area may be positive or negative - in accordance with the

// contour orientation

if (approx.size() == 4 &&

fabs(contourArea(Mat(approx))) > 1000 &&

isContourConvex(Mat(approx)))

{

double maxCosine = 0;


for (int j = 2; j < 5; j++)

{

// find the maximum cosine of the angle between joint edges

double cosine = fabs(angle(approx[j % 4], approx[j - 2], approx[j - 1]));

maxCosine = MAX(maxCosine, cosine);

}


// if cosines of all angles are small

// (all angles are ~90 degree) then write quandrange

// vertices to resultant sequence

if (maxCosine < 0.3)

squares.push_back(approx);

}

}

}

}

}

3.1硬件连接

    将摄像头与路由器连接,启动路由器,将PC连接到路由器的WIFI网络。

接线说明:

            ① 将2510通信转接板连接到扩展板的扩展坞上面;

            ② 用3根母对母杜邦线将2510通信转接板与WiFi路由器连接起来,GND-GND、RX-RX、TX-TX;

            ③ 找到1根USB线,一端连接到2510通信转接板接口上,另一端连接到WiFi路由器USB接口上;

            ④ 将摄像头线连接到WiFi路由器接口上。

3.2示例程序

    下面提供一个可以进行识别圆形和矩形的参考例程(ShapeDetect\ShapeDetect\MainWindow.xaml.cs):

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

using System.Threading.Tasks;

using System.Windows;

using System.Windows.Controls;

using System.Windows.Data;

using System.Windows.Documents;

using System.Windows.Input;

using System.Windows.Media;

using System.Windows.Media.Imaging;

using System.Windows.Navigation;

using System.Windows.Shapes;

using System.Windows.Forms;

using System.Runtime.InteropServices;

using System.Threading;


namespace ShapeDetect

{

    /// <summary>

    /// 形状识别

    /// </summary>

    public partial class MainWindow : Window

    {

        //定义检测模式

        int IMAGE_MODE = 1, VIDEO_MODE = 2;


        private AutoResetEvent exitEvent;

        private Thread m_thread;


        //导入动态链接库

        [DllImport("HoughCircles_DLL.dll")]

        //检测圆

        public static extern System.UIntPtr HoughCircles([MarshalAs(UnmanagedType.LPStr)]string address, int detect_mode);

        [DllImport("SquareDetect_DLL.dll")]

        //检测矩形

        public static extern void SquareDetector([MarshalAs(UnmanagedType.LPStr)]string address, int detect_mode);


        public MainWindow()

        {

            InitializeComponent();

        }


        private void Window_Loaded(object sender, RoutedEventArgs e)

        {

            GetIni();

            imgCheckBtn.IsChecked = true;

            circleCheckBtn.IsChecked = true;

        }


        //获取ini配置文件信息

        private void GetIni()

        {

            ini_RW.FileName = System.Windows.Forms.Application.StartupPath + "\\Config.ini";

            this.videoAddress.Text = ini_RW.ReadIni("VideoUrl", "videourl", "");

            this.ipAddress.Text = ini_RW.ReadIni("ControlUrl", "controlUrl", "");

            this.portBox.Text = ini_RW.ReadIni("ControlPort", "controlPort", "");

        }


        //修改配置

        private void setBtn_Click(object sender, RoutedEventArgs e)

        {

            ini_RW.WriteIni("VideoUrl", "videourl", this.videoAddress.Text);

            ini_RW.WriteIni("ControlUrl", "controlUrl", this.ipAddress.Text);

            ini_RW.WriteIni("ControlPort", "controlPort", this.portBox.Text);


            System.Windows.MessageBox.Show("配置成功!请重启程序以使配置生效。", "配置信息", MessageBoxButton.OK, MessageBoxImage.Information);

            //this.Close();

        }


        //计数清零

        private void BoxClean()

        {

            circleTextBox.Text = "0";

            recTextBox.Text = "0";

        }


        //打开图片地址

        private void imgBtn_Click(object sender, RoutedEventArgs e)

        {

            try

            {


                BoxClean();

                //WPF中,OpenFileDialog位于Microsoft.Win32名称空间

                Microsoft.Win32.OpenFileDialog dialog = new Microsoft.Win32.OpenFileDialog();

                dialog.Filter = "All files (*.*)|*.*|jpg files (*.jpg)|*.jpg|png files(*.png)|*.png";


                if (dialog.ShowDialog() == true)

                {


                    string path = dialog.FileName;

                    imgAddressBox.Text = path;

                }

            }

            catch { };

        }


        //检测形状判断

        private void ShapeDetect(string address, int mode)

        {

            if (circleCheckBtn.IsChecked == true)

            {

                //System.Windows.MessageBox.Show("检测圆形");

                circleTextBox.Text = HoughCircles(address, mode).ToString();

            }

            else if (recCheckBtn.IsChecked == true)

            {

                //System.Windows.MessageBox.Show("检测矩形");

                SquareDetector(address, mode);

            }

        }


        //图片检测

        private void imgDetect()

        {


            if (imgAddressBox.Text == string.Empty)

            {

                System.Windows.MessageBox.Show(

                    "图片地址为空,请选择一张图片", "警告",

                    MessageBoxButton.OK,

                    MessageBoxImage.Information

                    );

                return;

            }

            else

            {

                ShapeDetect(imgAddressBox.Text, IMAGE_MODE);

            }

        }


        //视频检测

        private void videoDetect()

        {

            try

            {

                while (true)

                {

                    this.Dispatcher.Invoke(

                        new Action(

                            delegate

                            {

                                string ip = this.videoAddress.Text;

                                ShapeDetect(ip, VIDEO_MODE);

                            }

                            ));

                }


            }

            catch { };

        }


        //判断检测为图片还是视频,开启形状检测

        private void detectBtn_Click(object sender, RoutedEventArgs e)

        {

            BoxClean();

            if (imgCheckBtn.IsChecked == true)

            {

                imgDetect();

            }

            else if (videoCheckBtn.IsChecked == true)

            {

                try

                {

                    m_thread = new Thread(new ThreadStart(videoDetect));

                    m_thread.Start();

                }

                catch { };

            }

        }


        //按esc键退出视频检测,结束线程

        private void Window_KeyDown(object sender, System.Windows.Input.KeyEventArgs e)

        {

            if (e.Key == Key.Escape)

            {

                exitEvent.Set();

                m_thread.Join();

            }

        }


    }

}

      程序识别圆形及矩形两种形状,包括对图像以及视频中的物体的形状检测,可参考上面的演示视频进行操作。图片中物体的形状识别,文末资料下载中提供一张测试图片,然后选择图片按钮,选择要检测的圆形或者矩形,点击形状检测。

      视频中的形状识别,选择视频按钮,选择要检测的圆形或者矩形,点击形状检测,可以使用球体或者矩形状物体进行检测。

4. 资料内容

①识别形状-例程源代码

②测试图片.jpg

资料内容详见 如何实现视觉识别-识别形状

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

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

相关文章

深入了解网络通信原理

同一个 vlan 同一个网段的 ip 地址能够直接互通&#xff0c;那么同一个 vlan 不同的网段能不能互通呢&#xff1f;海翎光电的小编整理了一篇文章&#xff0c;可以解决大家在网络中遇到的一些奇怪的问题&#xff0c;也是可以加深对网络的理解。 一、同 VLAN 不同网段能否 Ping 通…

Linux基本背景介绍与应用场景,Linux两条版本线,创建删除用户与修改密码的指令

TIPS Linux是一款叫做操作系统的软件&#xff0c;比如说你买了一个笔记本&#xff0c;你以为你只买了一个笔记本&#xff0c;实际上笔记本里面还搭建了一个非常重要的软件&#xff1a;操作系统。正是因为有了操作系统的存在&#xff0c;所以说有一个行为&#xff1a;开机就诞生…

再摘一枚重要奖项!腾讯安全获得云安全联盟CSA 2022安全金盾奖

4月13日&#xff0c;第六届云安全联盟大中华区大会&#xff08;CSA GCR Congress&#xff09;在上海举办&#xff0c;大会由联合国数字安全联盟、上海市经济和信息化委员会、上海市委网络安全和信息化委员会办公室、上海市普陀区人民政府指导&#xff0c;云安全联盟大中华区主办…

【开源项目】BallCat 项目脚手架

简介 &#x1f389;&#x1f389;&#x1f389; 基于 React 和 Ant Design 版本的前端 ballcat-ui-react 已发布&#xff0c;欢迎大家尝鲜使用 BallCat 组织旨在为项目快速开发提供一系列的基础能力&#xff0c;方便使用者根据项目需求快速进行功能拓展。 在以前使用其他后台管…

开源协议 GPL、BSD、MIT、Mozilla、Apache和LGPL的区别

一、GPL、BSD、MIT、Mozilla、Apache和LGPL的总体结构图如下&#xff1a; 二、协议详细说明 LGPL开源许可证&#xff1a; LGPL 是GPL的一个为主要为类库使用设计的开源协议。和GPL要求任何使用/修改/衍生之GPL类库的的软件必须采用GPL协议不同。LGPL 允许商业软件通过类库引用…

Python学习笔记--文件操作

&#xff08;一&#xff09; 文件的编码 1. 编码 编码是一种规则集合&#xff0c;记录了内容和二进制间进行相互转换的逻辑 编码有许多种&#xff0c;最常用的是utf-8 2. 使用编码的原因 计算机只能识别二进制数&#xff0c;因此要将文件的内容翻译为二进制数&#xff0c;才能保…

【K8S系列】深入解析有状态服务

目录 序言 1 基本介绍 2 使用介绍 2.1 Headless Service 2.2 PersistentVolume 2.3 StatefulSet 2.4 Init Containers 3 问题 4 投票 序言 在你想要放弃的时候&#xff0c;想想是什么让你当初坚持走到了这里。 Kubernetes (k8s) 是一个容器编排平台&#xff0c;允许在…

RISC-V Linux 编译设备树

RISC-V Linux 编译设备树 flyfish 设备树是一种描述硬件资源的数据结构 引入设备树的主要目的 曾经 Linux内核中夹杂着大量的设备信息&#xff0c;影响Linux驱动开发效率&#xff0c;因为外部设备发生任何改动&#xff0c;需要重新编写和编译驱动代码。 现在 设备驱动程…

VSCode使用Remote SSH远程连接Linux服务器【远程开发】

文章目录前言视频教程1、安装OpenSSH2、vscode配置ssh3. 局域网测试连接远程服务器4. 公网远程连接4.1 ubuntu安装cpolar4.2 创建隧道映射4.3 测试公网远程连接5. 配置固定TCP端口地址5.1 保留一个固定TCP端口地址5.2 配置固定TCP端口地址5.3 测试固定公网地址远程转发自CSDN远…

shell一些操作

文章目录【检查脚本目录】示例【时间和时区】修改时间修改时区修改时区查看时区脚本从服务器同步client主动同步client被动同步docker-compose 时区设置kubernetes设置时区【按时间备份】实例date命令【历史命令和快捷键】历史命令快捷键【重定向】相关符号脚本输出到黑洞cat输…

android studio 页面布局(1)

<?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:app"http://schemas.android.com/apk/res-auto"xmlns:tools"http://schemas.android.com/too…

论文阅读 - ANEMONE: Graph Anomaly Detection with Multi-Scale Contrastive Learning

目录 摘要 1 简介 2 问题陈述 3 PROPOSED ANEMONE FRAMEWORK 3.1 多尺度对比学习模型 3.1.1 增强的自我网络生成 3.1.2 补丁级对比网络 3.1.3 上下文级对比网络 3.1.4 联合训练 3.2 统计异常估计器 4 EXPERIMENTS 4.1 Experimental Setup 4.1.1 Datasets 4.1.2 …

Ethercat学习-QT添加SOEM主站

文章目录简介环境搭建1.QT安装2.VS安装3.Win10 Debuggers4.QT配置SOEM移植&#xff11;.lib库生成2.文件移植1.文件整理2.添加文件至QT工程测试简介 QT版本5.14.2 VS版本2017 系统Win10 环境搭建 环境搭建很简单&#xff0c;就是安装几个软件就好了&#xff0c;没有特殊的…

快速排序及优化

快速排序及优化 概要 关于快速排序的原理不赘述&#xff0c;可以查看912. 排序数组 - 力扣&#xff08;Leetcode&#xff09; 本篇文章旨在提供快速排序的C#实现&#xff0c;并通过随机pivot&#xff0c;三数取中&#xff0c;小区间使用插入排序&#xff0c;栈实现&#xff…

Makefile学习

什么是Makefile 使用 GCC 编译器在 Linux 进行 C 语言编译&#xff0c;通过在终端执行 gcc 命 令来完成 C 文件的编译&#xff0c;如果我们的工程只有一两个 C 文件还好&#xff0c;需要输入的命令不多&#xff0c;当文件有几十、上百甚至上万个的时候用终端输入 GCC 命令的方…

NIFI大数据进阶_实时同步MySql的数据到Hive中去_可增量同步_实时监控MySql数据库变化_操作方法说明_01---大数据之Nifi工作笔记0033

然后我们来看如何把mysql数据实时同步到hive中去 可以看到,其实就是使用 CaptureChangeMySql来获取mysql中变化的数据,具体就是增删改数据 然后再用RouteOnAttribute通过属性,也就是根据是增删改的话,根据这个属性进行路由 把数据路由到不同的位置 然后再用EvaluateJsonPa…

创客匠人人物访谈:客户,如何反推创业发展? |

古语有云&#xff1a;“君者&#xff0c;舟也&#xff1b;庶人者&#xff0c;水也&#xff1b;水则载舟&#xff0c;水则覆舟”。 从创业的角度来看&#xff0c;客户&#xff0c;就是创业这艘小船在市场的海洋里能够安稳航行的必要条件。 在创业的过程中&#xff0c;一定是先有…

STC89C52串口通信当中的UART

1.串口介绍 串口是一种应用十分广泛的通讯接口&#xff0c;串口成本低、容易使用、通信线路简单&#xff0c;可实现两个设备的互相通信。 单片机的串口可以使单片机与单片机、单片机与电脑、单片机与各式各样的模块互相通信&#xff0c;极大的扩展了单片机的应用范围&#xf…

Vue 也能复用模板了

文章目录Vue 也能复用模板了1. 怎么用1.1 基本使用1.2 传递参数2. 怎么实现的3. 注意事项4. 未来发展Vue 也能复用模板了 相信很多使用 Vue 的同学会遇到这样一个问题&#xff1a; “组件里有几个部分的模板&#xff08;html 结构&#xff09;是相同的&#xff0c;要想复用的…

使用线性回归构建波士顿房价预测模型

使用线性回归构建波士顿房价预测模型 描述 波士顿房价数据集统计了波士顿地区506套房屋的特征以及它们的成交价格&#xff0c;这些特征包括周边犯罪率、房间数量、房屋是否靠河、交通便利性、空气质量、房产税率、社区师生比例&#xff08;即教育水平&#xff09;、周边低收入…