主頁 >  其他 > opencv實戰——機器視覺檢測和計數

opencv實戰——機器視覺檢測和計數

2021-06-09 16:55:34 其他

引言

在機器視覺中,有時需要對產品進行檢測和計數,其難點無非是對于產品的影像分割,

由于之前網購的維生素片,有時候忘了今天有沒有吃過,就想對瓶子里的藥片計數...在學習opencv以后,希望實作對于維生素片分割計數演算法,本次實戰在基于形態學的基礎上又衍生出基于距離變換的分水嶺演算法,使其實作的效果更具普遍性,


基于形態學的維生素片檢測和計數

??整體思路:

  1. 讀取圖片
  2. 形態學處理(在二值化前進行適度形態學處理,效果俱佳)
  3. 二值化
  4. 提取輪廓(進行藥片分割)
  5. 獲取輪廓索引,并篩選所需要的輪廓
  6. 畫出輪廓,顯示計數

opencv實作:

int main(int argc, char** argv)
{
    Mat src, src_binary,dst,src_distance;
    src = imread("D:/opencv練習圖片/維生素片機器視覺檢測和計數.png");
    imshow("原圖片", src);
    Mat kernel = getStructuringElement(MORPH_RECT, Size(16, 16), Point(-1, -1));
    morphologyEx(src, dst, MORPH_OPEN, kernel);
    imshow("形態學",dst);
    cvtColor(dst, dst, COLOR_RGB2GRAY);
    threshold(dst, src_binary, 100, 255, THRESH_OTSU);
    imshow("二值化", src_binary);
    vector<vector<Point>> contours;
    findContours(src_binary, contours, RETR_EXTERNAL, CHAIN_APPROX_NONE, Point(0, 0));
    RNG rng(12345);
    double area;
    Point2i PL;
    for (size_t i = 0; i < contours.size(); i++)
    {
        area = contourArea(contours[i]);
        if (area < 500)continue;
        PL = contours[i].front();
        Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255));
        drawContours(src, contours, i, color, 2, 8);
        putText(src, to_string(i), PL, FONT_HERSHEY_COMPLEX, 1, color, 2);            
    }
    imshow("計數結果", src);
    waitKey(0);
    return 0;
}

效果展示:

 由上圖可以看的,原圖在經過形態學處理后,可以去除很多細節,簡化后續的藥片分割操作,

但是在計數結果圖上發現,索引17號藥片并沒有完全分割(實際上修改形態學的結構元素尺寸(改為20*20)也可以完全分離這兩個藥片),

??這不由得讓我們思考,如果簡單的形態學處理分割不了藥片呢?

 對于復雜的產品圖片,我們可以使用基于距離變換的分水嶺演算法對其分割,


 基于距離變換的分水嶺演算法檢測和計數

OpenCV 采用了基于標記點的分水嶺演算法,在這種演算法中我們要設定哪些山谷點會匯合,哪些不會,這是一種互動式的影像分割,我們要做的就是給我們已知的物件打上不同的標簽(即添加注水點),然后實施分水嶺演算法,每一次灌水,我們的標簽就會被更新,當兩個不同顏色的標簽相遇時就構建堤壩,直到將所有山峰淹沒,最后我們得到的邊界物件(堤壩)的值為 -1,

??對于如何打上標簽(即添加注水點)有兩種辦法:

opencv中,對于一張二值化的影像,后續處理方式有兩種,第一種方式就是利用findContours、drawContours等函式進行輪廓分析(opencv以對輪廓的處理為主),第二種方式就是計算連通域進行區域分析,

第一種(基于輪廓):在二值化后,對影像尋找輪廓findContours,篩選出注水區域輪廓,然后通過drawContours對輪廓標記,

第二種(基于區域):在二值化后,先對尋找影像中的前景圖(即注水點),再尋找到背景圖(進行膨脹),最后找到未知區域(背景減去前景,得到邊緣圖),通過connectedComponents()獲取標記點,

相關API:

  • 分水嶺函式watershed函式原型
void watershed( InputArray image, InputOutputArray markers );

第一個輸入引數 image,必須是CV_8UC3型別影像,

第二個輸入/輸出引數markers必須是32位單通道影像,和image尺寸一樣,包含不同區域的輪廓,每個輪廓有一個自己唯一的編號,

??在執行watershed函式后,演算法會根據markers傳入的輪廓作為種子,對影像上其他的像素點根據分水嶺演算法規則進行判斷,并對每個像素點的區域歸屬進行劃定,直到處理完影像上所有像素點,而區域與區域之間的分界處的值被置為“-1”,以做區分,

  •  距離變換函式distanceTransform函式原型

距離變換運算用于計算二值化影像中的每一個非零點距自己最近的零點的距離,距離變換影像上越亮的點,代表了這一點距離零點的距離越遠,

距離變換通常用于求解影像的骨骼和查找物體的質心(即獲取距離變換的極大值)和計算非零像素到最近零像素點的最短距離,

distanceTransform( InputArray src, OutputArray dst, int distanceType, int maskSize,int dstType = CV_32F);

第一個輸入引數src,必須是CV_8UC1型別的二值影像(只有0或1)

第二個輸出引數dst,表示的是計算距離的輸出影像,輸出型別是CV_32F/CV_8U的單通道影像,大小與輸入圖片相同,

第三個引數distanceType,表示的是選取距離的型別,可以設定為DIST_L1,DIST_L2,DIST_C

第四個引數maskSize,表示的是距離變換的掩膜模板,可以設定為3,5(常用3)

第四個引數dstType,表示輸出型別,可選擇CV_32F/CV_8U

注:若輸出型別為CV_32F,想要顯示距離變換后的骨架影像,需要對其歸一化,(normalize)

??????????????

先來看看第一種標記mark(基于輪廓)的方法:


 (一)讀入影像,形態學,二值化(消除噪聲)



Mat src, src_binary, dst, src_distance; src
= imread("D:/opencv練習圖片/維生素片機器視覺檢測和計數.png"); imshow("原圖片", src); Mat kernel = getStructuringElement(MORPH_RECT, Size(3, 3), Point(-1, -1)); morphologyEx(src, dst, MORPH_OPEN, kernel); imshow("形態學", dst); cvtColor(dst, dst, COLOR_RGB2GRAY); threshold(dst, src_binary, 100, 255, THRESH_OTSU); imshow("二值化", src_binary);


 (二)距離變換(歸一化顯示),再二值化


    distanceTransform(src_binary, src_distance, DIST_L2, 3, 5);
    normalize(src_distance, src_distance, 0, 1, NORM_MINMAX);
    imshow("距離變換", src_distance);
    threshold(src_distance, src_distance, 0.4,1, THRESH_BINARY);
    imshow("再二值化", src_distance);

 

經過距離變換后的二值化,可以清晰看到,藥片以及完全分割開來,


 (三)打上標簽(添加注水點),基于輪廓


    //尋找標記點marsk的輪廓資訊 也就是分水嶺的水壩
    src_distance.convertTo(src_distance, CV_8UC1);
    vector<vector<Point>> contours;    
    findContours(src_distance, contours, RETR_TREE, CHAIN_APPROX_SIMPLE);
    //創建maker
    Mat markers = Mat::zeros(src.size(), CV_32S);//  //因為分水嶺后的邊緣存盤是-1,所以必須使用有符號的CV_32S
    for (size_t t = 0; t < contours.size(); t++) 
    {
        drawContours(markers, contours, static_cast<int>(t), Scalar(static_cast<int>(t) + 1), -1);//輪廓數字編號
    }
    circle(markers, Point(5, 5), 30, Scalar(255), -1);//關鍵代碼(mark做一個小標記)
    int index1 = 0;
    //列印輪廓資料 有值的均為輪廓線
    for (int row = 0; row < markers.rows; row++)
        for (int col = 0; col < markers.cols; col++)
        {
            index1 = markers.at<int>(row, col);
            cout << index1 << ",";
        }

 部分標簽markers輪廓資料截圖,可以看到0代表背景,輪廓線用正數索引標識


 (四)進行分水嶺操作,并給分水嶺后的區域隨機上色,并列印出檢測的藥片個數,


 // 生成隨機顏色
    vector<Vec3b> colors;
    for (size_t i = 0; i < contours.size(); i++) {
        int r = theRNG().uniform(0, 255);
        int g = theRNG().uniform(0, 255);
        int b = theRNG().uniform(0, 255);
        colors.push_back(Vec3b((uchar)b, (uchar)g, (uchar)r));
    }

    // 顏色填充與最終顯示
    Mat dst1 = Mat::zeros(markers.size(), CV_8UC3);
    int index = 0;
    for (int row = 0; row < markers.rows; row++) {
        for (int col = 0; col < markers.cols; col++) {
            index = markers.at<int>(row, col);
            
            if (index > 0 && index <= contours.size()) {
                dst1.at<Vec3b>(row, col) = colors[index - 1];

            }
            else {
                dst1.at<Vec3b>(row, col) = Vec3b(0, 0, 0);
            }

        }
    }
    imshow("結果顯示", dst1);
    printf("藥片檢測個數: %d\n", contours.size());

 

 

 ??????????????

再來看看第二種標記mark(基于區域)的方法:


 (一)讀入影像,形態學,二值化(消除噪聲)


 Mat foreground, background, unkonwn;//創建前景,背景,未知區域
    Mat src, src_binary, dst, src_distance;
    src = imread("D:/opencv練習圖片/維生素片機器視覺檢測和計數.png");
    imshow("原圖片", src);
    Mat kernel = getStructuringElement(MORPH_RECT, Size(3, 3), Point(-1, -1));
    morphologyEx(src, dst, MORPH_OPEN, kernel);
    imshow("形態學", dst);
    cvtColor(dst, dst, COLOR_RGB2GRAY);
    threshold(dst, src_binary, 100, 255, THRESH_OTSU);
    imshow("二值化", src_binary);

 


(二)對二值化影像進行膨脹操作,得到大部分是背景的圖片


//得到背景圖片
    dilate(src_binary, background, kernel, Point(-1, -1), 3);
    imshow("背景圖片", background);


(三)通過對二值影像距離變換得到前景圖片(即注水點)


//距離變換
    distanceTransform(src_binary, src_distance, DIST_L2, 3, 5);
    imshow("距離變換", src_distance);
    normalize(src_distance, src_distance, 0, 255, NORM_MINMAX);
    double my_minv = 0.0, my_maxv = 0.0;
    minMaxIdx(src_binary, &my_minv, &my_maxv);
    threshold(src_distance, foreground, 0.4 * my_maxv, 255, THRESH_BINARY);
    foreground.convertTo(foreground, CV_8U);
    imshow("前景圖片", foreground);


 (四)通過背景與前景的差值,得到未知區域(即邊緣所在區域)


//得到未知區域
    unkonwn = background - foreground;
    imshow("未知區域", unkonwn);


 (五)得到這些區域以后,我們可以獲取注水點的標簽,通過connectedComponents實作(即獲取markers標簽)


//創建標記點markers
    Mat markers = Mat(src.size(), CV_32S);
    int num = connectedComponents(foreground, markers, 8);
    cout << num << endl;
    markers = markers + 1;
    for (int i = 0; i < unkonwn.rows; i++)
    {
        for (int j = 0; j < unkonwn.cols; j++)
        {
            if (((int)unkonwn.at<uchar>(i, j)) == 255)
            {
                markers.at<signed int>(i, j) = 0;
            }
        }
    }

詳細理解該步驟:

現在我們已經知道哪些是背景,哪些是藥片(前景區域),

因此我們可以創建一個標簽(和原圖大小,型別為CV_32S),通過connectedComponents函式對前景區域進行標記

連通域相關博文:opencv——連通域標記與分析 - 唯有自己強大 - 博客園 (cnblogs.com)

該函式會對前景區域連通域分析,并將背景設定為0,其他區域從1開始正整數標記(這就是我們的種子,水漫時會從這里漫出),結果回傳給markers,

但是對于分水嶺演算法,會將為0的區域認為是未知區域,因此要markers整體加一,


(六)進行分水嶺操作,并顯示邊緣


watershed(src, markers);
    for (int row = 0; row < markers.rows; row++)
    {
        for (int col = 0; col < markers.cols; col++)
        {

            if (markers.at< int>(row, col) == -1)
            {
                src.at<Vec3b>(row, col) = Vec3b(0, 0, 255);
            }
        }
    }

    imshow("結果", src);

 

由于分水嶺演算法會將找到的邊緣在markers置為-1,因此我們對原圖操作,將索引為-1的位置的像素值改為紅色(即顯示邊緣),

 

 參考鏈接:OpenCV---分水嶺演算法 - 山上有風景 - 博客園 (cnblogs.com)

                  (8條訊息) c++和opencv小知識:基于距離變換的分水嶺演算法(固定流程)_夢游城市的博客-CSDN博客

                  (8條訊息) OpenCV分水嶺演算法影像分割_冰冰bing的博客-CSDN博客

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/285709.html

標籤:其他

上一篇:直方圖統計和顯示

下一篇:Action Sheets 和 Activity Views

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more