色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Opencv計數程序詳解

新機器視覺 ? 來源:機器視覺沙龍 ? 2023-12-18 16:45 ? 次閱讀

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

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

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

整體思路:

讀取圖片

形態學處理(在二值化前進行適度形態學處理,效果俱佳)

二值化

提取輪廓(進行藥片分割)

獲取輪廓索引,并篩選所需要的輪廓

畫出輪廓,顯示計數

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> 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);


21ff5900-9d7a-11ee-8b88-92fbcf53809c.png221de7e4-9d7a-11ee-8b88-92fbcf53809c.png2238c7f8-9d7a-11ee-8b88-92fbcf53809c.png

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

 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);

22513c5c-9d7a-11ee-8b88-92fbcf53809c.png2265b89e-9d7a-11ee-8b88-92fbcf53809c.png

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

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

//尋找標記點marsk的輪廓信息 也就是分水嶺的水壩
    src_distance.convertTo(src_distance, CV_8UC1);
    vector> 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(t), Scalar(static_cast(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(row, col);
            cout << index1 << ",";
        }

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

2278b9b2-9d7a-11ee-8b88-92fbcf53809c.png

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

// 生成隨機顏色
    vector 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(row, col);
            
            if (index > 0 && index <= contours.size()) {
                dst1.at(row, col) = colors[index - 1];


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


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

22b5c366-9d7a-11ee-8b88-92fbcf53809c.png22c6c576-9d7a-11ee-8b88-92fbcf53809c.png

22d60e32-9d7a-11ee-8b88-92fbcf53809c.png

再來看看第二種標記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);

22dc9644-9d7a-11ee-8b88-92fbcf53809c.png

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

//距離變換
    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);

22e9f7bc-9d7a-11ee-8b88-92fbcf53809c.png

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

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

2308cdb8-9d7a-11ee-8b88-92fbcf53809c.png

(五)得到這些區域以后,我們可以獲取注水點的標簽,通過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(i, j)) == 255)
            {
                markers.at(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(row, col) = Vec3b(0, 0, 255);
            }
        }
    }


    imshow("結果", src);

23168bb0-9d7a-11ee-8b88-92fbcf53809c.png

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







審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 機器視覺
    +關注

    關注

    162

    文章

    4389

    瀏覽量

    120446
  • OpenCV
    +關注

    關注

    31

    文章

    635

    瀏覽量

    41388

原文標題:詳解Opencv計數程序

文章出處:【微信號:vision263com,微信公眾號:新機器視覺】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    OpenCV兩種不同方法實現粘連大米分割計數

    測試圖如下,圖中有個別米粒相互粘連,本文主要演示如何使用OpenCV用兩種不同方法將其分割并計數
    的頭像 發表于 01-22 14:55 ?1801次閱讀
    <b class='flag-5'>OpenCV</b>兩種不同方法實現粘連大米分割<b class='flag-5'>計數</b>

    RK3568 + OpenCV 會碰撞出什么火花?案例詳解:2-1 基于OpenCV的畫線實驗

    應用,都可以利用它來作開發; 所有API函數的源代碼都是公開的,可以看到其內部實現的程序步驟; 可以修改OpenCV的源代碼,編譯生成需要的特定API函數。 OpenCV處理圖像 功能:線段的繪制
    發表于 12-03 14:09

    如何使用SDK在獨立模式下的OpenCV應用程序

    嗨,大家好,我是新手SDK用戶。我使用vivado hls來合成一個簡單的圖像處理算法,并構建了我現在使用sdk的硬件平臺,我想使用sdk在獨立模式下使用opencv應用程序在zynq fpga上
    發表于 05-04 17:09

    OpenCV C++程序編譯與演示

    1、在JetsonNano上編譯OpenCV源碼與OpenCV C++ YOLOv5程序演示  編譯OpenCV最新4.5.x版本  Jetson Nano自帶的
    發表于 11-10 16:42

    播放視頻_OpenCV3版書本配套示例程序06

    OpenCV3編程入門》OpenCV3版書本配套示例程序06,使用VideoCapture類進行視頻讀取和播放。
    發表于 06-06 15:20 ?0次下載

    遍歷圖像像素的14種方法_OpenCV2版書本配套示例程序24

    遍歷圖像像素的14種方法_OpenCV2版書本配套示例程序24,來自一本國外OpenCV2書籍的示例-遍歷圖像像素的14種方法。
    發表于 06-06 15:20 ?0次下載

    OpenCV的Mat數據格式及其遍歷的程序資料說明

    opencv早期的版本中,圖像通過一個叫做IplImage的結構(structure)存儲在內存中。由于C語言對程序員高度的信任,因此它需要手動地對內存進行管理,比如內存的分配和回收,這在大型
    的頭像 發表于 02-17 09:42 ?4200次閱讀
    <b class='flag-5'>OpenCV</b>的Mat數據格式及其遍歷的<b class='flag-5'>程序</b>資料說明

    OpenCV進行橢圓擬合的程序免費下載

    本文檔的主要內容詳細介紹的是OpenCV進行橢圓擬合的程序免費下載。
    發表于 10-12 14:58 ?4次下載

    OpenCV的混合高斯背景模型源碼程序免費下載

    本文檔的主要內容詳細介紹的使用OpenCV的混合高斯背景模型源碼程序免費下載
    發表于 10-18 11:55 ?5次下載

    如何使用Borland C++ Builder6.0來開發OpenCV程序

    本文檔的主要內容詳細介紹的是如何使用Borland C++ Builder6.0來開發OpenCV程序
    發表于 05-26 17:32 ?14次下載
    如何使用Borland C++ Builder6.0來開發<b class='flag-5'>OpenCV</b>的<b class='flag-5'>程序</b>

    BorlandCBuilder6.0安裝OPENCV方法

    BorlandCBuilder6.0安裝OPENCV方法(新型電源技術結課論文UC3842)-文檔為BorlandCBuilder6.0安裝OPENCV方法詳解文檔,是一份不錯的參考資料,感興趣的可以下載看看,,,,,,,,,,
    發表于 09-17 15:34 ?9次下載
    BorlandCBuilder6.0安裝<b class='flag-5'>OPENCV</b>方法

    使用Raspberry Pi構建一個OpenCV人群計數裝置

    在本教程中,我們將使用 Raspberry Pi 和 ThingSpeak 構建一個 OpenCV 人群計數
    的頭像 發表于 08-12 17:24 ?3041次閱讀
    使用Raspberry Pi構建一個<b class='flag-5'>OpenCV</b>人群<b class='flag-5'>計數</b>裝置

    在JetsonNano上編譯OpenCV源碼與OpenCV C++ YOLOv5程序演示

    Jetson Nano自帶的OpenCV版本比較低,Jetpack4.6對應的OpenCV版本為4.1的,有圖為證。
    的頭像 發表于 11-10 11:28 ?3305次閱讀

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

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

    opencv-python和opencv一樣嗎

    不一樣。OpenCV(Open Source Computer Vision Library)是一個開源的計算機視覺和機器學習軟件庫,它提供了大量的圖像和視頻處理功能。OpenCV
    的頭像 發表于 07-16 10:38 ?1267次閱讀
    主站蜘蛛池模板: 国产精品99久久久久久AV色戒| 国产成人欧美日韩在线电影| 91福利在线观看| 在线精品视频免费观看| 国产电影午夜成年免费视频| 青青青青青青草| 99久久精品免费精品国产| 噜噜噜狠狠夜夜躁精品| 在线少女漫画| 里番※琉璃全彩acg奈亚子| 伊人久久国产免费观看视频| 精品免费久久久久久成人影院| 亚洲免费视频观看| 久久re视频这里精品一本到99| 亚洲午夜精品aaa级久久久久| 黄色xxxxxx| 这里只有精品网| 高h 大尺度纯肉 np快穿| 色噜噜噜视频| 国产精品久久久久永久免费看 | 成人精品在线视频| 日韩视频在线观看| 影视先锋男人无码在线| 久久久GOGO无码啪啪艺术| 最新国产精品福利2020| 暖暖日本在线手机免费完整版| youjizz护士| 香港日本三级亚洲三级| 精品人妻无码一区二区三区蜜桃臀| 亚洲中字幕永久在线观看| 两个人看的www免费高清直播| a级精品九九九大片免费看| 日产久久视频| 国产精品嫩草影院一区二区三区| 亚洲国产在线精品国自产拍五月| 九九热在线免费观看| 99精品无码AV在线播放| 涩涩免费网站| 久草在线福利资站免费视频| 99国产精品免费视频| 涩涩网站在线看|