首頁 > 軟體

opencv對多種顏色小球的形狀及位置判斷方式

2022-11-11 14:00:46

一、opencv是什麼?

OpenCV是一個用於影象處理、分析、機器視覺方面的開源函數庫.

二、使用步驟

1.引入庫

程式碼如下:

import cv2
import numpy as np

2.設定顏色閾值

程式碼如下:

#顏色閾值
low_red = np.array([100, 100, 60])
up_red = np.array([180, 255, 255])
low_green = np.array([35, 43, 46])
up_green = np.array([77, 255, 255])
low_blue = np.array([90, 110, 110])
up_blue = np.array([124, 255, 255])
#記錄形狀
xz = {}

字典xz是待會記錄形狀用的

3.對圖片進行載入和處理

程式碼如下:

if __name__ == '__main__':
    image = cv2.imread('img_1.png')#讀取影象
    image = cv2.resize(image, (500, 500))#重新裁剪影象
    #image = cv2.GaussianBlur(image, (11, 11), 0)#高斯濾波
    hsv = cv2.cvtColor(image, cv2.COLOR_BGR2HSV)
    color_read(hsv, image)#處理圖片的函數
    cv2.imshow('image', image)
    cv2.waitKey(0)
    cv2.destroyAllWindows()

4.處理圖片的函數

程式碼如下:

def color_read(hsv, image):
    global xz
    blue = get_image(hsv, low_blue, up_blue)#使用獲取顏色空間的函數進行顏色獲取三種顏色
    red = get_image(hsv, low_red, up_red)
    green = get_image(hsv, low_green, up_green)
    mask = blue + red + green
    #尋找影象的輪廓
    cnts = cv2.findContours(mask.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)[-2]
    print('cnts:', len(cnts))
    ##先判斷形狀再畫輪廓
    for cnt in cnts:
        area = cv2.contourArea(cnt)#計算輪廓的面積
        print('area:', area)
        if area > 1000:
            epsilon = 0.04 * cv2.arcLength(cnt, True)#計算輪廓長度
            approx = cv2.approxPolyDP(cnt, epsilon, True)#計算輪廓角點
            corners = len(approx)
            print(corners)
            #根據角點的個數判斷形狀
            if corners == 3:
                a = '三角形'
                b = approx[0][0][0]  # 根據三角形的角點判斷位置
                xz[a] = b
                area, trgl = cv2.minEnclosingTriangle(cnt)#尋找三角形的輪廓
                # 繪製三角形輪廓
                for i in range(0, 3):
                    cv2.line(image, tuple(trgl[i][0]), tuple(trgl[(i + 1) % 3][0]), (0, 255, 0), 2)
            elif corners == 4:
                a = '矩形'
                b = approx[0][0][0]
                xz[a] = b
                x, y, w, h = cv2.boundingRect(cnt)#尋找矩形輪廓
                cv2.rectangle(image, (x, y), (x + w, y + h), (0, 255, 0), 2)#繪製矩形輪廓
                
            else:
            	#圓形這裡是處理的不好的地方所以我沒有用變數x1和z,因為畫出來的輪廓有點大
                a = '圓'
                b = approx[0][0][0]
                xz[a] = b
                ((x1, y1), z) = cv2.minEnclosingCircle(cnt)#尋找圓形輪廓
                x1 = int(x1)
                y1 = int(y1)
                z = int(z)
                cv2.circle(image, (150, y1), 60, (0, 255, 0), 2)#繪製圓形輪廓

5.獲取顏色空間函數

程式碼如下:

#獲取影象hsv的方法
def get_image(hsv, low, up):
    mask = cv2.inRange(hsv, low, up)#獲取色彩空間
    mask = cv2.erode(mask, None, 2)#腐蝕操作
    mask = cv2.dilate(mask, None, 2)#膨脹操作
    return mask

6.執行效果

圖片之前拍的不是很好所以我截成了小圖,圖片拍的是幾何圖所以效果不太好,如果是平面的可能會好一點,圖二是列印字典xz的輸出。

7.完整程式碼

#獲取影象hsv的方法
import cv2
import numpy as np
#顏色閾值
low_red = np.array([100, 100, 60])
up_red = np.array([180, 255, 255])
low_green = np.array([35, 43, 46])
up_green = np.array([77, 255, 255])
low_blue = np.array([90, 110, 110])
up_blue = np.array([124, 255, 255])
#記錄形狀
xz = {}

#獲取影象hsv的方法
def get_image(hsv, low, up):
    mask = cv2.inRange(hsv, low, up)#獲取色彩空間
    mask = cv2.erode(mask, None, 2)#腐蝕操作
    mask = cv2.dilate(mask, None, 2)#膨脹操作
    return mask

#獲取輪廓
def color_read(hsv, image):
    global xz
    blue = get_image(hsv, low_blue, up_blue)#使用獲取顏色空間的函數進行顏色獲取三種顏色
    red = get_image(hsv, low_red, up_red)
    green = get_image(hsv, low_green, up_green)
    mask = blue + red + green
    #尋找影象的輪廓
    cnts = cv2.findContours(mask.copy(), cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)[-2]
    print('cnts:', len(cnts))
    ##先判斷形狀再畫輪廓
    for cnt in cnts:
        area = cv2.contourArea(cnt)#計算輪廓的面積
        print('area:', area)
        if area > 1000:
            epsilon = 0.04 * cv2.arcLength(cnt, True)#計算輪廓長度
            approx = cv2.approxPolyDP(cnt, epsilon, True)#計算輪廓角點
            corners = len(approx)
            print(corners)
            #根據角點的個數判斷形狀
            if corners == 3:
                a = '三角形'
                b = approx[0][0][0]  # 根據三角形的角點判斷位置
                xz[a] = b
                area, trgl = cv2.minEnclosingTriangle(cnt)#尋找三角形的輪廓
                # 繪製三角形輪廓
                for i in range(0, 3):
                    cv2.line(image, tuple(trgl[i][0]), tuple(trgl[(i + 1) % 3][0]), (0, 255, 0), 2)
            elif corners == 4:
                a = '矩形'
                b = approx[0][0][0]
                xz[a] = b
                x, y, w, h = cv2.boundingRect(cnt)#尋找矩形輪廓
                cv2.rectangle(image, (x, y), (x + w, y + h), (0, 255, 0), 2)#繪製矩形輪廓

            else:
                a = '圓'
                b = approx[0][0][0]
                xz[a] = b
                ((x1, y1), z) = cv2.minEnclosingCircle(cnt)#尋找圓形輪廓
                x1 = int(x1)
                y1 = int(y1)
                z = int(z)
                cv2.circle(image, (150, y1), 60, (0, 255, 0), 2)#繪製圓形輪廓


if __name__ == '__main__':
    image = cv2.imread('img_1.png')#讀取影象
    image = cv2.resize(image, (500, 500))#重新裁剪影象
    #image = cv2.GaussianBlur(image, (11, 11), 0)#高斯濾波
    hsv = cv2.cvtColor(image, cv2.COLOR_BGR2HSV)
    color_read(hsv, image)#處理圖片的函數
    cv2.imshow('image', image)
    cv2.waitKey(0)
    cv2.destroyAllWindows()
    #對拿到的小球字典位置資料進行處理
    xz[min(xz, key=xz.get)] = 1
    xz[max(xz, key=xz.get)] = 3
    xz[max(xz, key=xz.get)] = 2
    xz = sorted(xz.items(), key=lambda x: x[1], reverse=False)
    xz = dict(xz)

    print(xz)

總結

這是這個程式碼的詳細和功能介紹,希望能給大家一個參考,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com