OpenCV 360度行车辅助系统——红绿灯智能预判

2024-01-25 01:30

本文主要是介绍OpenCV 360度行车辅助系统——红绿灯智能预判,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  1. AI学习红绿灯

此功能仅限前置摄像头。当视频刚开始播放时,通过自研AI算法学习红绿像素高频出现的位置,学习结束后根据学习结果判断是否使用AI红绿灯 区域,并在接下来的视频部分针对区域进行红绿灯识别。

检测到红灯则在视频底端显示“Red”字样,检测到绿灯则在视频底端显示“Green”字样。

 

 

1、收集前5%帧的红绿色素点

2、按照坐标大小排序,删掉前后10%,删掉异常大、小的值

3、筛选后判断结果数量是否充足,充足则开始使用,不足则用系统默认的区域

4、在应用的过程中持续监测红绿像素,如果出现大幅度像素密集点偏移,则重新学习

Mat player::RG_detector(Mat frame)
{Mat detected_frame = frame.clone();Mat roi(detected_frame, Rect(400,200,400,100));Rect detected_rect(400,200,400,100);//调整亮度frame.convertTo(img, img.type(), a, b);//转换为YCrCb颜色空间cvtColor(img, imgYCrCb, CV_BGR2YCrCb);imgRed.create(imgYCrCb.rows, imgYCrCb.cols, CV_8UC1);imgGreen.create(imgYCrCb.rows, imgYCrCb.cols, CV_8UC1);//分解YCrCb的三个成分vector<Mat> planes;split(imgYCrCb, planes);// 遍历以根据Cr分量拆分红色和绿色MatIterator_<uchar> it_Cr = planes[1].begin<uchar>(),it_Cr_end = planes[1].end<uchar>();MatIterator_<uchar> it_Red = imgRed.begin<uchar>();MatIterator_<uchar> it_Green = imgGreen.begin<uchar>();for (; it_Cr != it_Cr_end; ++it_Cr, ++it_Red, ++it_Green){// RED, 145<Cr<470if (*it_Cr > 145 && *it_Cr < 470)*it_Red = 255;else*it_Red = 0;// GREEN 95<Cr<110if (*it_Cr > 95 && *it_Cr < 110)*it_Green = 255;else*it_Green = 0;}redCount = processImgR(imgRed);greenCount = processImgG(imgGreen);cout << "red:" << redCount << ";  " << "green:" << greenCount << endl;if(currentFrameNumber < 50){putText(frame, "AI learning", hintTextPoint, font_face, font_scale, Scalar(135, 74, 32), thickness, 8, 0);}else{if(redCount > greenCount ){putText(frame, "Red", hintTextPoint, font_face, font_scale, Scalar(41, 41, 239), thickness, 8, 0);}else if(redCount < greenCount ){putText(frame, "Green", hintTextPoint, font_face, font_scale, Scalar(22, 210, 115), thickness, 8, 0);}}return frame;
}bool player::RGB_AI_learn_x_y(int x, int y)
{if(currentFrameNumber <= totalFrameNumber / 20){if(y <= frame.cols/2){RGB_learn_array_x.push_back(x);RGB_learn_array_y.push_back(y);}}else{sort(RGB_learn_array_x.begin(), RGB_learn_array_x.end());sort(RGB_learn_array_y.begin(), RGB_learn_array_y.end());if(int subArrayNum = RGB_learn_array_x.size() - RGB_learn_array_y.size() != 0){//删除多余元素if(subArrayNum > 0){for(int i=0; i<subArrayNum; i++){RGB_learn_array_x.pop_back();}}else if(subArrayNum < 0){for(int i=0; i>subArrayNum; i--){RGB_learn_array_y.pop_back();}}}int delete_count = 0, delete_total = RGB_learn_array_x.size()/10;for(delete_count = 0; delete_count < delete_total; delete_count++){//删除 10% 较大值RGB_learn_array_x.pop_back();RGB_learn_array_y.pop_back();}sort(RGB_learn_array_x.rbegin(), RGB_learn_array_x.rend());sort(RGB_learn_array_y.rbegin(), RGB_learn_array_y.rend());for(delete_count = 0; delete_count < delete_total; delete_count++){//删除 10% 较小值RGB_learn_array_x.pop_back();RGB_learn_array_y.pop_back();}sort(RGB_learn_array_x.begin(), RGB_learn_array_x.end());sort(RGB_learn_array_y.begin(), RGB_learn_array_y.end());long long int sum_x, sum_y;sum_x = accumulate(begin(RGB_learn_array_x),end(RGB_learn_array_x), 0);sum_y = accumulate(begin(RGB_learn_array_y),end(RGB_learn_array_y), 0);//计算均值RGB_learn_x = (int)sum_x / RGB_learn_array_x.size();RGB_learn_y = (int)sum_y / RGB_learn_array_y.size();//计算范围RGB_learn_w = RGB_learn_array_x.end() - RGB_learn_array_x.begin();RGB_learn_h = RGB_learn_array_y.end() - RGB_learn_array_y.begin();if(RGB_learn_array_x.size()>50)return true;//如果学习到大量红绿灯,认为学习有效else return false;}
}int player::processImgR(Mat src)
{Mat tmp;vector<vector<Point>> contours;vector<Vec4i> hierarchy;vector<Point> hull;CvPoint2D32f tempNode;CvMemStorage* storage = cvCreateMemStorage();CvSeq* pointSeq = cvCreateSeq(CV_32FC2, sizeof(CvSeq), sizeof(CvPoint2D32f), storage);Rect* trackBox;Rect* result;int resultNum = 0;int area = 0;src.copyTo(tmp);//提取轮廓findContours(tmp, contours, hierarchy, CV_RETR_CCOMP, CV_CHAIN_APPROX_SIMPLE);if (contours.size() > 0){trackBox = new Rect[contours.size()];result = new Rect[contours.size()];//确定要跟踪的区域for (int i = 0; i < contours.size(); i++){cvClearSeq(pointSeq);// 获取凸包的点集convexHull(Mat(contours[i]), hull, true);int hullcount = (int)hull.size();// 凸包的保存点for (int j = 0; j < hullcount - 1; j++){//if(HoughCircle_detector(src) > 1){bool final_learn_result;if(currentFrameNumber <= totalFrameNumber / 20){//前面 5% 帧用来学习红绿灯位置final_learn_result = RGB_AI_learn_x_y(hull[j].x, hull[j].y);}else{if(final_learn_result == true){//学习有效,使用AI识别红绿灯if(hull[j].x >= RGB_learn_x-RGB_learn_w/2 && hull[j].x <= RGB_learn_x+RGB_learn_w/2 &&hull[j].y >= RGB_learn_y-RGB_learn_h/2 && hull[j].y <= RGB_learn_y+RGB_learn_h/2){//cout<<"AI success!"<<endl;tempNode.x = hull[j].x;tempNode.y = hull[j].y;cvSeqPush(pointSeq, &tempNode);}}else{if(hull[j].x >= 400 && hull[j].x <= 800 &&hull[j].y >= 400 && hull[j].y <= 550){tempNode.x = hull[j].x;tempNode.y = hull[j].y;cvSeqPush(pointSeq, &tempNode);}}}}}trackBox[i] = cvBoundingRect(pointSeq);}if (isFirstDetectedR){lastTrackBoxR = new Rect[contours.size()];for (int i = 0; i < contours.size(); i++)lastTrackBoxR[i] = trackBox[i];lastTrackNumR = contours.size();isFirstDetectedR = false;}else{for (int i = 0; i < contours.size(); i++){for (int j = 0; j < lastTrackNumR; j++){if (isIntersected(trackBox[i], lastTrackBoxR[j])){result[resultNum] = trackBox[i];break;}}resultNum++;}delete[] lastTrackBoxR;lastTrackBoxR = new Rect[contours.size()];for (int i = 0; i < contours.size(); i++){lastTrackBoxR[i] = trackBox[i];}lastTrackNumR = contours.size();}delete[] trackBox;}else{isFirstDetectedR = true;result = NULL;}cvReleaseMemStorage(&storage);if (result != NULL){for (int i = 0; i < resultNum; i++){area += result[i].area();}}delete[] result;return area;
}int player::processImgG(Mat src)
{Mat tmp;vector<vector<Point> > contours;vector<Vec4i> hierarchy;vector< Point > hull;CvPoint2D32f tempNode;CvMemStorage* storage = cvCreateMemStorage();CvSeq* pointSeq = cvCreateSeq(CV_32FC2, sizeof(CvSeq), sizeof(CvPoint2D32f), storage);Rect* trackBox;Rect* result;int resultNum = 0;int area = 0;src.copyTo(tmp);// Extract the contourfindContours(tmp, contours, hierarchy, CV_RETR_CCOMP, CV_CHAIN_APPROX_SIMPLE);if (contours.size() > 0){trackBox = new Rect[contours.size()];result = new Rect[contours.size()];// Determine the area to trackfor (int i = 0; i < contours.size(); i++){cvClearSeq(pointSeq);// Get the point set of the convex hullconvexHull(Mat(contours[i]), hull, true);int hullcount = (int)hull.size();// Save points of the convex hullfor (int j = 0; j < hullcount - 1; j++){//if(HoughCircle_detector(src) > 1){bool final_learn_result;if(currentFrameNumber <= totalFrameNumber / 20){//前面 5% 帧用来学习红绿灯位置final_learn_result = RGB_AI_learn_x_y(hull[j].x, hull[j].y);}else{if(final_learn_result == true){//学习有效,使用AI识别红绿灯if(hull[j].x >= RGB_learn_x-RGB_learn_w/2 && hull[j].x <= RGB_learn_x+RGB_learn_w/2 &&hull[j].y >= RGB_learn_y-RGB_learn_h/2 && hull[j].y <= RGB_learn_y+RGB_learn_h/2){//cout<<"AI success!"<<endl;tempNode.x = hull[j].x;tempNode.y = hull[j].y;cvSeqPush(pointSeq, &tempNode);}}else{if(hull[j].x >= 400 && hull[j].x <= 800 &&hull[j].y >= 400 && hull[j].y <= 550){tempNode.x = hull[j].x;tempNode.y = hull[j].y;cvSeqPush(pointSeq, &tempNode);}}}}}trackBox[i] = cvBoundingRect(pointSeq);}if (isFirstDetectedG){lastTrackBoxG = new Rect[contours.size()];for (int i = 0; i < contours.size(); i++)lastTrackBoxG[i] = trackBox[i];lastTrackNumG = contours.size();isFirstDetectedG = false;}else{for (int i = 0; i < contours.size(); i++){for (int j = 0; j < lastTrackNumG; j++){if (isIntersected(trackBox[i], lastTrackBoxG[j])){result[resultNum] = trackBox[i];break;}}resultNum++;}delete[] lastTrackBoxG;lastTrackBoxG = new Rect[contours.size()];for (int i = 0; i < contours.size(); i++){lastTrackBoxG[i] = trackBox[i];}lastTrackNumG = contours.size();}delete[] trackBox;}else{isFirstDetectedG = true;result = NULL;}cvReleaseMemStorage(&storage);if (result != NULL){for (int i = 0; i < resultNum; i++){area += result[i].area();}}delete[] result;return area;
}

这篇关于OpenCV 360度行车辅助系统——红绿灯智能预判的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/641669

相关文章

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

基于Python实现自动化邮件发送系统的完整指南

《基于Python实现自动化邮件发送系统的完整指南》在现代软件开发和自动化流程中,邮件通知是一个常见且实用的功能,无论是用于发送报告、告警信息还是用户提醒,通过Python实现自动化的邮件发送功能都能... 目录一、前言:二、项目概述三、配置文件 `.env` 解析四、代码结构解析1. 导入模块2. 加载环

linux系统上安装JDK8全过程

《linux系统上安装JDK8全过程》文章介绍安装JDK的必要性及Linux下JDK8的安装步骤,包括卸载旧版本、下载解压、配置环境变量等,强调开发需JDK,运行可选JRE,现JDK已集成JRE... 目录为什么要安装jdk?1.查看linux系统是否有自带的jdk:2.下载jdk压缩包2.解压3.配置环境

Linux查询服务器系统版本号的多种方法

《Linux查询服务器系统版本号的多种方法》在Linux系统管理和维护工作中,了解当前操作系统的版本信息是最基础也是最重要的操作之一,系统版本不仅关系到软件兼容性、安全更新策略,还直接影响到故障排查和... 目录一、引言:系统版本查询的重要性二、基础命令解析:cat /etc/Centos-release详

更改linux系统的默认Python版本方式

《更改linux系统的默认Python版本方式》通过删除原Python软链接并创建指向python3.6的新链接,可切换系统默认Python版本,需注意版本冲突、环境混乱及维护问题,建议使用pyenv... 目录更改系统的默认python版本软链接软链接的特点创建软链接的命令使用场景注意事项总结更改系统的默

在Linux系统上连接GitHub的方法步骤(适用2025年)

《在Linux系统上连接GitHub的方法步骤(适用2025年)》在2025年,使用Linux系统连接GitHub的推荐方式是通过SSH(SecureShell)协议进行身份验证,这种方式不仅安全,还... 目录步骤一:检查并安装 Git步骤二:生成 SSH 密钥步骤三:将 SSH 公钥添加到 github

使用Python构建智能BAT文件生成器的完美解决方案

《使用Python构建智能BAT文件生成器的完美解决方案》这篇文章主要为大家详细介绍了如何使用wxPython构建一个智能的BAT文件生成器,它不仅能够为Python脚本生成启动脚本,还提供了完整的文... 目录引言运行效果图项目背景与需求分析核心需求技术选型核心功能实现1. 数据库设计2. 界面布局设计3

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

Linux系统之lvcreate命令使用解读

《Linux系统之lvcreate命令使用解读》lvcreate是LVM中创建逻辑卷的核心命令,支持线性、条带化、RAID、镜像、快照、瘦池和缓存池等多种类型,实现灵活存储资源管理,需注意空间分配、R... 目录lvcreate命令详解一、命令概述二、语法格式三、核心功能四、选项详解五、使用示例1. 创建逻

OpenCV在Java中的完整集成指南分享

《OpenCV在Java中的完整集成指南分享》本文详解了在Java中集成OpenCV的方法,涵盖jar包导入、dll配置、JNI路径设置及跨平台兼容性处理,提供了图像处理、特征检测、实时视频分析等应用... 目录1. OpenCV简介与应用领域1.1 OpenCV的诞生与发展1.2 OpenCV的应用领域2