數字圖像課程工程大作業分析

試題分析:
在連續的視頻中對火焰及水柱的軌跡檢測,效果如圖。
在這里插入圖片描述
**
提示:
1、火焰可利用亮度和顏色
2、水柱的軌跡需要先用背景差分獲得水柱的連通域,然后利用連通域上的像素點進行曲線的擬合,水槍的位置視為已知,即可以手動活動坐標。**

1、火焰檢測

我們先截一張圖,觀察HSV三個通道圖長什么樣子:

觀察之后決定從S通道著手,首先確定火焰的位置是固定不變的(總是在右下角)
對S通道的圖片進行二值化獲得二值圖,再通過限制像素位置,排除干擾:

在這里插入圖片描述
對灰度圖進行二值化,發現限制200時,字的影響消失
在這里插入圖片描述

//識別并標出火焰
//輸入 原圖像 輸出:原圖像上框出框框(火)
void find_fire(Mat& srcMat, Mat& outputMat)
{Mat gray_srcMat;Mat dstMat, binMat;cvtColor(srcMat, gray_srcMat, COLOR_BGR2GRAY);Mat fire_Mat = Mat::zeros(gray_srcMat.size(), gray_srcMat.type());cvtColor(srcMat, dstMat, COLOR_BGR2HSV);vector<Mat> channels;split(dstMat, channels);//將S通道的圖像復制,然后處理Mat S_Mat;channels.at(1).copyTo(S_Mat);int row_num = srcMat.rows;			//行數int col_num = srcMat.cols;			//列數//雙重循環,遍歷右下角像素值for (int i = row_num * 0.75;i < row_num;i++)	//行循環{for (int j = col_num * 0.75;j < col_num;j++)	//列循環{//-------【開始處理每個像素】---------------if ((gray_srcMat.at<uchar>(i, j) >= 150 && S_Mat.at<uchar>(i, j) >= 120)){fire_Mat.at<uchar>(i, j) = 255;}//-------【處理結束】---------------}}//膨脹Mat element = getStructuringElement(MORPH_RECT, Size(15, 15));	//返回的是內核矩陣Mat firedstImage;dilate(fire_Mat, fire_Mat, element);//通過findContours函數尋找連通域vector<vector<Point>> contours;vector<Vec4i> hierarchy;findContours(fire_Mat, contours, RETR_LIST, CHAIN_APPROX_NONE);//繪制輪廓for (int i = 0; i < contours.size(); i++) {RotatedRect rbox = minAreaRect(contours[i]);drawContours(srcMat, contours, i, Scalar(0, 255, 255), 1, 8);Point2f vtx[4];rbox.points(vtx);for (int j = 0; j < 4; ++j) {cv::line(outputMat, vtx[j], vtx[j < 3 ? j + 1 : 0], Scalar(255, 255, 255), 2, LINE_AA);}}
}

2、水柱檢測

利用最簡單的背景差分(幀差法),獲得圖像:
1
框出來的地方是誤差,我們對其進行排除

//輸入:圖片 輸出:去除干擾后的圖
void clear_other_disturb(Mat& srcMa)
{//通過觀察、發現0-180列為干擾,320-480列且0-90行為干擾for (int i = 0;i < 90;i++)	//行循環{for (int j = 320;j < srcMa.cols;j++)	//列循環{//-------【開始處理每個像素】---------------srcMa.at<uchar>(i, j) = 0;//-------【處理結束】---------------}}for (int i = 0;i < srcMa.rows;i++)	//行循環{for (int j = 0;j < 180;j++)	//列循環{//-------【開始處理每個像素】---------------srcMa.at<uchar>(i, j) = 0;//-------【處理結束】---------------}}
}

在這里插入圖片描述
二值化一下:
在這里插入圖片描述
接下來選擇擬合的點:
連通域中的像素坐標:

//輸入:二值化圖片  輸出:點集 (每列最多只有一個點,符合函數一一映射的關系)
void find_point(Mat& binMat, std::vector<cv::Point>& key_point)
{//第一個點肯定是水槍位置//遍歷:170列到430列//		30行到230行//遍歷方式:遍歷每列,從上往下遍歷(小到大),找到的第一個為白的點記錄下來,如果遍歷完這一列并沒有發現白色點,則不作記錄//注意這里的循環方式與其他地方的相反for (int i = 170;i < 430;i++)	//列循環{for (int j =30;j < 230;j++)	//行循環{if (binMat.at<uchar>(j, i) == 255){//行列與坐標系對應關系//行rows: Y(height)//列cols : X(width)//注意!注意!注意!//在Mat類型變量訪問時下標是反著寫的key_point.push_back(cv::Point(i,j));break;}}}//打印出來讓我看看int N= key_point.size();for (int n = 0;n < N;n++){cout << "point"<< key_point[n].x<<" "<< key_point[n].y << endl;}//畫出來Mat image = cv::Mat::zeros(binMat.rows, binMat.cols, CV_8UC3);for (int i = 0; i < key_point.size(); i++){cv::circle(image, key_point[i], 5, cv::Scalar(0, 0, 255), 2, 8, 0);}imshow("tu", image);
}

將選出的點畫出來:
在這里插入圖片描述
將像素點選取出來之后,接下來就是用最小二乘法擬合曲線,具體內容講解請看:
https://blog.csdn.net/guduruyu/article/details/72866144

//輸入:待擬合的點集,擬合出來的曲線的點
bool polynomial_curve_fit(std::vector<cv::Point>& key_point, int n, cv::Mat& A)
{//Number of key pointsint N = key_point.size();//構造矩陣Xcv::Mat X = cv::Mat::zeros(n + 1, n + 1, CV_64FC1);for (int i = 0; i < n + 1; i++){for (int j = 0; j < n + 1; j++){for (int k = 0; k < N; k++){X.at<double>(i, j) = X.at<double>(i, j) +std::pow(key_point[k].x, i + j);}}}//構造矩陣Ycv::Mat Y = cv::Mat::zeros(n + 1, 1, CV_64FC1);for (int i = 0; i < n + 1; i++){for (int k = 0; k < N; k++){Y.at<double>(i, 0) = Y.at<double>(i, 0) +std::pow(key_point[k].x, i) * key_point[k].y;}}A = cv::Mat::zeros(n + 1, 1, CV_64FC1);//求解矩陣Acv::solve(X, Y, A, cv::DECOMP_LU);return true;
}

最終代碼:

#include <opencv2/opencv.hpp>
#include "opencv2/features2d.hpp"
#include <iostream>
#include "windows.h"
#include <stdio.h>
#include <time.h>
#include <math.h>  
#include<opencv2/imgproc/types_c.h> 
//#include "My_ImageProssing_base.h"
#define WINDOW_NAME1 "【程序窗口1】"			
#define WINDOW_NAME2 "【程序窗口2】"	
using namespace cv;
using namespace std;
RNG g_rng(12345);//識別并標出火焰
//輸入 原圖像 輸出:原圖像上框出框框(火)
void find_fire(Mat& srcMat, Mat& outputMat)
{Mat gray_srcMat;Mat dstMat, binMat;cvtColor(srcMat, gray_srcMat, COLOR_BGR2GRAY);Mat fire_Mat = Mat::zeros(gray_srcMat.size(), gray_srcMat.type());cvtColor(srcMat, dstMat, COLOR_BGR2HSV);vector<Mat> channels;split(dstMat, channels);//將S通道的圖像復制,然后處理Mat S_Mat;channels.at(1).copyTo(S_Mat);int row_num = srcMat.rows;			//行數int col_num = srcMat.cols;			//列數//雙重循環,遍歷右下角像素值for (int i = row_num * 0.75;i < row_num;i++)	//行循環{for (int j = col_num * 0.75;j < col_num;j++)	//列循環{//-------【開始處理每個像素】---------------if ((gray_srcMat.at<uchar>(i, j) >= 150 && S_Mat.at<uchar>(i, j) >= 120)){fire_Mat.at<uchar>(i, j) = 255;}//-------【處理結束】---------------}}//膨脹Mat element = getStructuringElement(MORPH_RECT, Size(15, 15));	//返回的是內核矩陣Mat firedstImage;dilate(fire_Mat, fire_Mat, element);//通過findContours函數尋找連通域vector<vector<Point>> contours;vector<Vec4i> hierarchy;findContours(fire_Mat, contours, RETR_LIST, CHAIN_APPROX_NONE);//繪制輪廓for (int i = 0; i < contours.size(); i++) {RotatedRect rbox = minAreaRect(contours[i]);drawContours(srcMat, contours, i, Scalar(0, 255, 255), 1, 8);Point2f vtx[4];rbox.points(vtx);for (int j = 0; j < 4; ++j) {cv::line(outputMat, vtx[j], vtx[j < 3 ? j + 1 : 0], Scalar(255, 255, 255), 2, LINE_AA);}}
}
//--------------------------------------------工程大作業---------------------------------------------------//輸入:二值化圖片  輸出:點集 (每列最多只有一個點,符合函數一一映射的關系)
void find_point(Mat& binMat, std::vector<cv::Point>& key_point)
{//第一個點肯定是水槍位置//遍歷:170列到430列//		30行到230行//遍歷方式:遍歷每列,從上往下遍歷(小到大),找到的第一個為白的點記錄下來,如果遍歷完這一列并沒有發現白色點,則不作記錄//注意這里的循環方式與其他地方的相反for (int i = 170;i < 430;i++)	//列循環{for (int j =30;j < 230;j++)	//行循環{if (binMat.at<uchar>(j, i) == 255){//行列與坐標系對應關系//行rows: Y(height)//列cols : X(width)//注意!注意!注意!//在Mat類型變量訪問時下標是反著寫的key_point.push_back(cv::Point(i,j));break;}}}//打印出來讓我看看int N= key_point.size();for (int n = 0;n < N;n++){cout << "point"<< key_point[n].x<<" "<< key_point[n].y << endl;}//畫出來Mat image = cv::Mat::zeros(binMat.rows, binMat.cols, CV_8UC3);for (int i = 0; i < key_point.size(); i++){cv::circle(image, key_point[i], 5, cv::Scalar(0, 0, 255), 2, 8, 0);}//imshow("tu", image);
}//輸入:待擬合的點集,擬合出來的曲線的點
bool polynomial_curve_fit(std::vector<cv::Point>& key_point, int n, cv::Mat& A)
{//Number of key pointsint N = key_point.size();//構造矩陣Xcv::Mat X = cv::Mat::zeros(n + 1, n + 1, CV_64FC1);for (int i = 0; i < n + 1; i++){for (int j = 0; j < n + 1; j++){for (int k = 0; k < N; k++){X.at<double>(i, j) = X.at<double>(i, j) +std::pow(key_point[k].x, i + j);}}}//構造矩陣Ycv::Mat Y = cv::Mat::zeros(n + 1, 1, CV_64FC1);for (int i = 0; i < n + 1; i++){for (int k = 0; k < N; k++){Y.at<double>(i, 0) = Y.at<double>(i, 0) +std::pow(key_point[k].x, i) * key_point[k].y;}}A = cv::Mat::zeros(n + 1, 1, CV_64FC1);//求解矩陣Acv::solve(X, Y, A, cv::DECOMP_LU);return true;
}//輸入:圖片 輸出:去除干擾后的圖
void clear_other_disturb(Mat& srcMa)
{//通過觀察、發現0-180列為干擾,320-480列且0-90行為干擾for (int i = 0;i < 90;i++)	//行循環{for (int j = 320;j < srcMa.cols;j++)	//列循環{//-------【開始處理每個像素】---------------srcMa.at<uchar>(i, j) = 0;//-------【處理結束】---------------}}for (int i = 0;i < srcMa.rows;i++)	//行循環{for (int j = 0;j < 180;j++)	//列循環{//-------【開始處理每個像素】---------------srcMa.at<uchar>(i, j) = 0;//-------【處理結束】---------------}}
}
int main()
{//實例化的同時初始化VideoCapture capture("D:\\opencv_picture_test\\videos\\大作業.avi");		//類似于 int a=1;//如果視頻打開失敗//計數器int cnt = 0;Mat frame;Mat rgb_mat;		Mat rgb_mat1;Mat bgMat1;		//第0幀圖像Mat bgMat2;		//之后每幀圖像while (1){capture >> frame;//圖像保護if (!frame.data){cout << "src image load failed!" << endl;break;}//將彩色圖保存下來,用于最后畫圖的底布rgb_mat = frame.clone();rgb_mat1= frame.clone();//轉灰度圖,用于之后的找水柱cvtColor(frame, frame, COLOR_BGR2GRAY);if (cnt == 0){//第一幀,獲得背景圖像frame.copyTo(bgMat1);//找火find_fire(rgb_mat, rgb_mat);imshow("output", rgb_mat);waitKey(20);}else if (cnt<=100)	//水柱還沒有出現,我們不找水柱,防止誤判(其實是有誤判的),此時我們只繪制火焰{//找火find_fire(rgb_mat, rgb_mat);imshow("output", rgb_mat);waitKey(20);}else{//獲取本次場景的圖像frame.copyTo(bgMat2);//====================================進行處理================================================Mat result = bgMat1.clone();//【1】背景圖像和當前圖像相減absdiff(bgMat1, bgMat2, result);//【2】接下來是去除干擾clear_other_disturb(result);//【3】二值化一下threshold(result, result, 100, 255, THRESH_BINARY);//imshow("result", result);//【4】找點std::vector<cv::Point> points;find_point(result, points);//【5】找到點之后就是擬合曲線,這里采用【https://blog.csdn.net/guduruyu/article/details/72866144】的思路cv::Mat A;//這里使用3次函數,效果很好polynomial_curve_fit(points, 3, A);std::cout << "A = " << A << std::endl;std::vector<cv::Point> points_fitted;for (int x = 170; x < 430; x++){double y = A.at<double>(0, 0) + A.at<double>(1, 0) * x +A.at<double>(2, 0) * std::pow(x, 2) + A.at<double>(3, 0) * std::pow(x, 3);points_fitted.push_back(cv::Point(x, y));}//【6】在彩色圖上繪制水柱cv::polylines(rgb_mat, points_fitted, false, cv::Scalar(0, 255, 255), 1, 8, 0);find_fire(rgb_mat1, rgb_mat);imshow("output", rgb_mat);//====================================處理結束================================================waitKey(20);}cnt++;}waitKey(0);return 0;
}

本文章代碼可以隨意使用!

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/378478.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/378478.shtml
英文地址,請注明出處:http://en.pswp.cn/news/378478.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

設計電子商務網站的10個技巧(轉自ITEye)

導讀&#xff1a;隨著先進科學技術的應用&#xff0c;人們無需外出逛幾個小時來“獵”東西&#xff0c;直接坐在家里就可以購買所需商品&#xff0c;支付服務費用。你只需一臺電腦就能搞定。人們習慣了周到的服務和漂亮的櫥窗&#xff0c;對網店的選擇也不例外。因此&#xff0…

分析酸對酸性染料染羊毛染色性能的影響?舉例說明酸性染料染羊毛時,如何選擇合適的染浴pH值?并說明原因。

分析酸對酸性染料染羊毛染色性能的影響?舉例說明酸性染料染羊毛時,如何選擇合適的染浴pH值?并說明原因。 標準答案: 羊毛屬于蛋白質纖維,屬于兩性纖維,酸影響羊毛的解離程度,帶電性及帶電量,影響熱力學性能。等電點以下,羊毛帶正電荷,與陰離子酸性染料之間存在靜電引…

ORACLE連接數據庫(備忘)

常用命令&#xff1a; conn sys/密碼 as sysdba 連接數據庫轉載于:https://www.cnblogs.com/jiangu66/archive/2013/05/01/3053787.html

stl swap函數_vector :: swap()函數以及C ++ STL中的示例

stl swap函數C vector :: swap()函數 (C vector::swap() function) vector::swap() is a library function of "vector" header, it is used to swap the content of the vectors, it is called with a vector and accepts another vector as an argument and swaps…

C++語法:vector的使用

【1】vector的創建與元素插入【2】vector元素的訪問【3】vector的基本使用技巧【4】vector的幾個重要操作【1】vector的創建與元素插入 std::vector<cv::Point> points; //vector容器中保存的類型是Point for (int i 0;i < 10;i) {float x rng.uniform(0, img.cols…

一、經含氟防水劑整理的織物主要存在的不足?

經含氟防水劑整理的織物主要存在的不足? 收集資料階段 含氟防水劑有哪些優缺點 一、含氟防水劑的優點 1、防水效果好,等級高。而無氟防水劑效果相對來說要差一些; 2、兼具防油的功能。無氟防水劑是不具備防油功能的; 3、穩定性好、與其他助劑的配伍好,工藝適應性強;有機…

Apache Web Login Authentication

Apache Web Login Authentication: Adding password protection to a web site using Apache web server authentication. AuthLDAPURL ldap://ldap.your-domain.com:389/ostooges?uid?subAuthLDAPBindDN "cnStoogeAdmin,ostooges"AuthLDAPBindPassword secret1Aut…

oracle中的with的用法,oracle中with子句的用法(轉)

語法&#xff1a;WITH query_name AS (subquery)[, query_name AS (subquery) ]...使用在主select關鍵字前&#xff0c;oracle將其當做一個內聯視圖或者臨時表使用。例子&#xff1a;1.最簡單的使用方法&#xff1a;如查詢部門名稱包含“A”的所有員工信息--with clausewith a …

模擬一個排隊系統

現在有一個數據源&#xff0c;有兩種狀態&#xff08;ON OFF&#xff09;&#xff0c;ON的持續時間服從均值為T_on的指數分布&#xff0c;OFF的持續時間服從均值為T_off的指數分布&#xff0c;源只在ON的時候產生數據包&#xff0c;服從均值為λ的指數分布 模擬一個排隊系統 每…

stl中copy()函數_std :: copy_if()函數以及C ++ STL中的示例

stl中copy()函數C STL std :: copy_if()函數 (C STL std::copy_if() function) copy_if() function is a library function of algorithm header, it is used to copy the elements of a container, it copies the certain elements (which satisfy the given condition) of a…

C++語法:求vector中的最大值及其位置

代碼&#xff1a; #include <iostream> #include <vector> #include <algorithm> using namespace std;int main(){vector<int> a { 2,4,6,7,1,0,8,9,6,3,2 };auto maxPosition max_element(a.begin(), a.end());cout << *maxPosition <&l…

二、織物具備超級防水效果的條件?

織物具備超級防水效果的條件? 收集資料階段 蓮花效應(Lotus Effect),指蓮葉表面具有超疏水(superhydrophobicity)以及自潔(self-cleaning)的特性。 由于蓮葉具有疏水、不吸水的表面,落在葉面上的雨水會因表面張力的作用形成水珠,換言之,水與葉面的接觸角(contacta…

C#編碼簡單性之函數篇(如何編寫簡短的C#代碼,隨時更新)

作者&#xff1a;陳勇出處&#xff1a;blog.csdn.net/cheny_com這是編碼簡單性系列中的其中一篇&#xff0c;之前幾篇包括代碼篇和語義篇。因為要積累案例&#xff0c;會隨時更新。之前提到&#xff1a;編碼簡單性的“心法”就是&#xff1a;只要屏幕上有任何兩部分代碼看上去相…

R學習筆記(1):R是什么

本文最新版已更新至http://thinkinside.tk/2013/05/03/r_notes_1_what.html 在學習量化投資的時候&#xff0c;我發現了R&#xff08;www.r-project.org&#xff09;。R到底是什么呢&#xff1f;在開始之前&#xff0c;先看看R的神奇之處。 1. R初窺 從CRAN&#xff08;The Co…

oracle網卡,Oracle_bond網卡配置

***************************loyu*******************************************************雙網卡創建bond虛擬網卡實驗*************************cat > /etc/sysconfig/network-scripts/ifcfg-bond0 << EofDEVICEbond0BOOTPROTOstaticONBOOTyesIPADDR172.16.116.60N…

Python | 展示一個break語句示例

break is a keyword in python just like another programming language and it is used to break the execution of loop statement. 就像另一種編程語言一樣&#xff0c; break是python中的關鍵字&#xff0c;用于中斷loop語句的執行。 In the given example, loop is runni…

數字圖像處理知識總結

一&#xff1a;基本概念 數字圖像&#xff1a;指由被稱作像素的小塊區域組成的二維矩陣。將物理圖像行列劃分后&#xff0c;每個小塊區域稱為像素&#xff08;pixel&#xff09;。每個像素包括兩個屬性&#xff1a;位置和灰度。圖像數字化一般分為采樣、量化與編碼三個步驟。數…

三、“滌綸纖維和棉纖維兩組分纖維在滌/棉混紡織物燃燒過程中有著明顯的物理相互作用和化學相互作用”,解釋這兩種作用。

“滌綸纖維和棉纖維兩組分纖維在滌/棉混紡織物燃燒過程中有著明顯的物理相互作用和化學相互作用”,解釋這兩種作用。 收集資料階段 棉纖維燃燒后炭化,而滌綸燃燒時熔融滴落,由于棉纖維成為支持體,可使熔融纖維聚集,并阻止其滴落,使熔融纖維燃燒更加劇烈,即所謂"支架效應…

關于性能測試的通俗解釋

關于性能測試的通俗解釋: http://www.docin.com/p-645879730.html 轉載于:https://www.cnblogs.com/preftest/archive/2013/05/03/3057231.html

oracle marley,滾石雜志500大專輯,對歐美音樂感興趣的可以找來聽聽。

滾石雜志于2003年11月評選出的滾石雜志五百大專輯。值得一提的是&#xff0c;披頭士樂隊占據了前五中的三席&#xff0c;前五十中的7席&#xff0c;正式出版的專輯幾乎全部入選前五百。排名 演唱者 專輯001 披頭士樂隊(The Beatles) Sgt. Peppers Lonely Hearts Club Band002 海…