主頁 > 後端開發 > Python從零到壹丨帶你了解影像直方圖理論知識和繪制實作

Python從零到壹丨帶你了解影像直方圖理論知識和繪制實作

2023-05-18 07:40:23 後端開發

摘要:本文將從OpenCV和Matplotlib兩個方面介紹如何繪制直方圖,這將為影像處理像素對比提供有效支撐,

本文分享自華為云社區《[Python從零到壹] 五十.影像增強及運算篇之影像直方圖理論知識和繪制實作》,作者:eastmount,

一.影像直方圖理論知識

灰度直方圖是灰度級的函式,描述的是影像中每種灰度級像素的個數,反映影像中每種灰度出現的頻率,假設存在一幅6×6像素的影像,接著統計其1至6灰度級的出現頻率,并繪制如圖1所示的柱狀圖,其中橫坐標表示灰度級,縱坐標表示灰度級出現的頻率[1-2],

如果灰度級為0-255(最小值0為黑色,最大值255為白色),同樣可以繪制對應的直方圖,如圖2所示,左邊是一幅灰度影像(Lena灰度圖),右邊是對應各像素點的灰度級頻率,

為了讓影像各灰度級的出現頻數形成固定標準的形式,可以通過歸一化方法對影像直方圖進行處理,將待處理的原始影像轉換成相應的標準形式[3],假設變數r表示影像中像素灰度級,歸一化處理后會將r限定在下述范圍:

在灰度級中,r為0時表示黑色,r為1時表示白色,對于一幅給定影像,每個像素值位于[0,1]區間之內,接著計算原始影像的灰度分布,用概率密度函式P?實作,為了更好地進行數字影像處理,必須引入離散形式,在離散形式下,用rk表示離散灰度級,P(rk)代替P?,并滿足公式(2),

公式中,nk為影像中出現rk這種灰度的像素數,n是影像中像素總數,是概率論中的頻數,l是灰度級總數(通常l為256級灰度),接著在直角坐標系中做出rk和P(rk)的關系圖,則成為灰度級的直方圖[4],

假設存在一幅3×3像素的影像,其像素值如公式(3)所示,則歸一化直方圖的步驟如下:

首先統計各灰度級對應的像素個數,用x陣列統計像素點的灰度級,y陣列統計具有該灰度級的像素個數,其中,灰度為1的像素共3個,灰度為2的像素共1個,灰度為3的像素共2個,灰度為4的像素共1個,灰度為5的像素共2個,

接著統計總像素個數,如公式(5)所示,

最后統計各灰度級的出現概率,通過公式(6)進行計算,其結果如下:

繪制的歸一化圖行如圖3所示,橫坐標表示影像中各個像素點的灰度級,縱坐標表示出現這個灰度級的概率,

直方圖被廣泛應用于計算機視覺領域,在使用邊緣和顏色確定物體邊界時,通過直方圖能更好地選擇邊界閾值,進行閾值化處理,同時,直方圖對物體與背景有較強對比的景物的分割特別有用,可以應用于檢測視頻中場景的變換及影像中的興趣點,

二.OpenCV繪制直方圖

首先講解使用OpenCV庫繪制直方圖的方法,在OpenCV中可以使用calcHist()函式計算直方圖,計算完成之后采用OpenCV中的繪圖函式,如繪制矩形的rectangle()函式,繪制線段的line()函式來完成,其中,cv2.calcHist()的函式原型及常見六個引數如下:

hist = cv2.calcHist(images, channels, mask, histSize, ranges, accumulate)

  • hist表示直方圖,回傳一個二維陣列
  • images表示輸入的原始影像
  • channels表示指定通道,通道編號需要使用中括號,輸入影像是灰度影像時,它的值為[0],彩色影像則為[0]、[1]、[2],分別表示藍色(B)、綠色(G)、紅色(R)
  • mask表示可選的操作掩碼,如果要統計整幅影像的直方圖,則該值為None;如果要統計影像的某一部分直方圖時,需要掩碼來計算
  • histSize表示灰度級的個數,需要使用中括號,比如[256]
  • ranges表示像素值范圍,比如[0, 255]
  • accumulate表示累計疊加標識,默認為false,如果被設定為true,則直方圖在開始分配時不會被清零,該引數允許從多個物件中計算單個直方圖,或者用于實時更新直方圖;多個直方圖的累積結果用于對一組影像的直方圖計算

接下來的代碼是計算影像各灰度級的大小、形狀及頻數,接著呼叫plot()函式繪制直方圖曲線,

# -*- coding: utf-8 -*-
# By:Eastmount
import cv2  
import numpy as np
import matplotlib.pyplot as plt
import matplotlib
#讀取影像
src = cv2.imread('lena-hd.png')
#計算256灰度級的影像直方圖
hist = cv2.calcHist([src], [0], None, [256], [0,255])
#輸出直方圖大小、形狀、數量
print(hist.size)
print(hist.shape)
print(hist)
#設定字體
matplotlib.rcParams['font.sans-serif']=['SimHei']
#顯示原始影像和繪制的直方圖
plt.subplot(121)
plt.imshow(src, 'gray')
plt.axis('off')
plt.title("(a)Lena灰度影像")
plt.subplot(122)
plt.plot(hist, color='r')
plt.xlabel("x")
plt.ylabel("y")
plt.title("(b)直方圖曲線")
plt.show()

上述代碼繪制的“Lena”灰度影像所對應的直方圖曲線如圖4所示,圖4(a)表示原影像,圖4(b)表示對應的灰度直方圖曲線,

同時輸出直方圖的大小、形狀及數量,如下所示:

256
(256L, 1L)
[[7.000e+00]
 [1.000e+00]
 [0.000e+00]
 [6.000e+00]
 [2.000e+00]
 ....
 [1.000e+00]
 [3.000e+00]
 [2.000e+00]
 [1.000e+00]
 [0.000e+00]]

彩色影像呼叫OpenCV繪制直方圖的演算法與灰度影像一樣,只是從B、G、R三個放量分別進行計算及繪制,具體代碼如下所示,

# -*- coding: utf-8 -*-
# By:Eastmount
import cv2  
import numpy as np
import matplotlib.pyplot as plt
import matplotlib
#讀取影像
src = cv2.imread('lena.png')
#轉換為RGB影像
img_rgb = cv2.cvtColor(src, cv2.COLOR_BGR2RGB)
#計算直方圖
histb = cv2.calcHist([src], [0], None, [256], [0,255])
histg = cv2.calcHist([src], [1], None, [256], [0,255])
histr = cv2.calcHist([src], [2], None, [256], [0,255])
#設定字體
matplotlib.rcParams['font.sans-serif']=['SimHei']
#顯示原始影像和繪制的直方圖
plt.subplot(121)
plt.imshow(img_rgb, 'gray')
plt.axis('off')
plt.title("(a)Lena原始影像")
plt.subplot(122)
plt.plot(histb, color='b')
plt.plot(histg, color='g')
plt.plot(histr, color='r')
plt.xlabel("x")
plt.ylabel("y")
plt.title("(b)直方圖曲線")
plt.show()

最侄訓制的“Lena”彩色影像及其對應的彩色直方圖曲線如圖5所示,其中圖5(a)表示Lena原始影像,圖5(b)表示對應的彩色直方圖曲線,

三.Matplotlib繪制直方圖

Matplotlib是Python強大的資料可視化工具,主要用于繪制各種2D圖形,本小節Python繪制直方圖主要呼叫matplotlib.pyplot庫中hist()函式實作,它會根據資料源和像素級繪制直方圖,其函式主要包括五個常用的引數,如下所示:

n, bins, patches = plt.hist(arr, bins=50, normed=1, facecolor=‘green’, alpha=0.75)

  • arr表示需要計算直方圖的一維陣列
  • bins表示直方圖顯示的柱數,可選項,默認值為10
  • normed表示是否將得到的直方圖進行向量歸一化處理,默認值為0
  • facecolor表示直方圖顏色
  • alpha表示透明度
  • n為回傳值,表示直方圖向量
  • bins為回傳值,表示各個bin的區間范圍
  • patches為回傳值,表示回傳每個bin里面包含的資料,是一個串列

影像直方圖的Python實作代碼如下所示,該示例主要是通過matplotlib.pyplot庫中的hist()函式繪制的,注意,讀取的“lena-hd.png”影像的像素為二維陣列,而hist()函式的資料源必須是一維陣列,通常需要通過函式ravel()拉直影像,

# -*- coding: utf-8 -*-
# By:Eastmount
import cv2  
import numpy as np
import matplotlib.pyplot as plt
#讀取影像
src = cv2.imread('lena-hd.png')
#繪制直方圖
plt.hist(src.ravel(), 256)
plt.xlabel("x")
plt.ylabel("y")
plt.show()
#顯示原始影像
cv2.imshow("src", src)
cv2.waitKey(0)
cv2.destroyAllWindows()

讀取顯示的“lena”灰度影像如圖6所示,

最終的灰度直方圖如圖7所示,它將Lena圖256級灰度和各個灰度級的頻數繪制出來,其中x軸表示影像的256級灰度,y軸表示各個灰度級的頻數,

如果呼叫下列函式,則繪制的直方圖是經過標準化處理,并且顏色為綠色、透明度為0.75的直方圖,如圖8所示,

plt.hist(src.ravel(), bins=256, density=1, facecolor=‘green’, alpha=0.75)

彩色直方圖是高維直方圖的特例,它統計彩色圖片RGB各分量出現的頻率,即彩色概率分布資訊,彩色圖片的直方圖和灰度直方圖一樣,只是分別畫出三個通道的直方圖,然后再進行疊加,其代碼如下所示,Lena彩色原始影像如圖9所示,

# -*- coding: utf-8 -*-
# By:Eastmount
import cv2  
import numpy as np
import matplotlib.pyplot as plt
#讀取影像
src = cv2.imread('Lena.png')
#獲取BGR三個通道的像素值
b, g, r = cv2.split(src)
#繪制直方圖
plt.figure("Lena")
#藍色分量
plt.hist(b.ravel(), bins=256, density=1, facecolor='b', edgecolor='b', alpha=0.75)
#綠色分量
plt.hist(g.ravel(), bins=256, density=1, facecolor='g', edgecolor='g', alpha=0.75)
#紅色分量
plt.hist(r.ravel(), bins=256, density=1, facecolor='r', edgecolor='r', alpha=0.75)
plt.xlabel("x")
plt.ylabel("y")
plt.show()
#顯示原始影像
cv2.imshow("src", src)
cv2.waitKey(0)
cv2.destroyAllWindows()

繪制的彩色直方圖如圖10所示,包括紅色、綠色、藍色三種對比,

如果希望將三個顏色分量的柱狀圖分開繪制并進行對比,則使用下面的代碼實作,呼叫plt.figure(figsize=(8, 6))函式繪制視窗,以及plt.subplot()函式分別繪制4個子圖,

# -*- coding: utf-8 -*-
# By:Eastmount
import cv2  
import numpy as np
import matplotlib.pyplot as plt
import matplotlib
#讀取影像
src = cv2.imread('lena.png')
#轉換為RGB影像
img_rgb = cv2.cvtColor(src, cv2.COLOR_BGR2RGB)
#獲取BGR三個通道的像素值
b, g, r = cv2.split(src)
print(r,g,b)
plt.figure(figsize=(8, 6))
#設定字體
matplotlib.rcParams['font.sans-serif']=['SimHei']
#原始影像
plt.subplot(221)
plt.imshow(img_rgb)
plt.axis('off')
plt.title("(a)原影像")
#繪制藍色分量直方圖
plt.subplot(222)
plt.hist(b.ravel(), bins=256, density=1, facecolor='b', edgecolor='b', alpha=0.75)
plt.xlabel("x")
plt.ylabel("y")
plt.title("(b)藍色分量直方圖")
#繪制綠色分量直方圖
plt.subplot(223)
plt.hist(g.ravel(), bins=256, density=1, facecolor='g', edgecolor='g', alpha=0.75)
plt.xlabel("x")
plt.ylabel("y")
plt.title("(c)綠色分量直方圖")
#繪制紅色分量直方圖
plt.subplot(224)
plt.hist(r.ravel(), bins=256, density=1, facecolor='r', edgecolor='r', alpha=0.75)
plt.xlabel("x")
plt.ylabel("y")
plt.title("(d)紅色分量直方圖")
plt.show()

最終輸出的圖形如圖11所示,,圖11(a)表示原影像,圖11(b)表示藍色分量直方圖,圖11?表示綠色分量直方圖,圖11(d)表示紅色分類直方圖,

四.總結

本文主要講解影像直方圖理論知識以及直方圖繪制方法,并且包括Matplotlib和OpenCV兩種統計及繪制方法,灰度直方圖是灰度級的函式,描述的是影像中每種灰度級像素的個數,反映影像中每種灰度出現的頻率,這篇文章的知識點將為后續影像處理和影像運算對比提供支撐,

參考文獻:

  • [1] 岡薩雷斯. 數字影像處理(第3版)[M]. 北京:電子工業出版社, 2013.
  • [2] 張恒博, 歐宗瑛. 一種基于色彩和灰度直方圖的影像檢索方法[J]. 計算機工程, 2004.
  • [3] Eastmount. [數字影像處理] 四.MFC對話框繪制灰度直方圖[EB/OL]. (2015-05-31). https://blog.csdn.net/eastmount/article/details/46237463.
  • [4] 阮秋琦. 數字影像處理學(第3版)[M]. 北京:電子工業出版社, 2008.
  • [5] Eastmount. [Python影像處理] 十一.灰度直方圖概念及OpenCV繪制直方圖[EB/OL]. (2018-11-06). https://blog.csdn.net/Eastmount/article/details/83758402.

 

 

點擊關注,第一時間了解華為云新鮮技術~

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

標籤:其他

上一篇:java設計模式【抽象工廠模式】

下一篇:返回列表

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

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Python從零到壹丨帶你了解影像直方圖理論知識和繪制實作

    摘要:本文將從OpenCV和Matplotlib兩個方面介紹如何繪制直方圖,這將為影像處理像素對比提供有效支撐。 本文分享自華為云社區《[Python從零到壹] 五十.影像增強及運算篇之影像直方圖理論知識和繪制實作》,作者:eastmount。 一.影像直方圖理論知識 灰度直方圖是灰度級的函式,描述 ......

    uj5u.com 2023-05-18 07:40:23 more
  • java設計模式【抽象工廠模式】

    java設計模式【抽象工廠模式】 抽象工廠模式 抽象工廠模式是對簡單工廠模式的一個變種,它允許通過一個統一的介面來創建不同的產品實體,而無需指定具體的子類。在這個模式中,我們只關心產品的抽象介面,而將具體的產品實作留給子類去實作。這樣,我們可以通過創建不同的工廠物件來創建不同的產品實體,而無需關心它 ......

    uj5u.com 2023-05-18 07:40:11 more
  • web應用模式、API介面、介面測驗工具postman、如何在瀏覽器中測

    一、web應用模式 Django框架就是一種web框架,專門用來寫web專案,之前學的,寫的BBS專案,圖書管理系統,用的都是前后端混合開發 -后端人員,寫后端,也要寫【模板語法】 》xx.html的python代碼 -全堆疊開發-->前后端混合時代,比較多 從今天開始,學的是前后端分離 -后端人員, ......

    uj5u.com 2023-05-18 07:39:52 more
  • Django authenticate() 函式查找不到與提交的用戶名和密碼匹配的

    在你的user APP下面添加一個utils.py檔案 class UsernameMobileBackend(ModelBackend): def authenticate(self, request, username=None, password=None, **kwargs): """ 重寫 ......

    uj5u.com 2023-05-18 07:39:40 more
  • 檔案館庫房環境溫濕度一體化建設平臺之10款八防硬體設備詳細介紹

    檔案八防十防常用的十款設備 序號 名稱 1 溫濕度傳感器 2 空氣質量云測儀 3 恒濕凈化一體機 4 健康防護一體機 5 綜合智能觸摸一體化區域控制器 6 空調紅外學習控制模塊 7 漏水檢測控制器及感應線 8 數字煙霧傳感器 9 紅外防盜傳感器 10 系統軟體平臺 附件 檔案十防要求 1.溫濕度傳感 ......

    uj5u.com 2023-05-18 07:39:20 more
  • < Python全景系列-3 > Python控制流程盤點及高級用法、神秘技巧

    全面深入地介紹 Python 的控制流程,包括條件陳述句、回圈結構和例外處理等關鍵部分,尤其會將串列決議、生成器、裝飾器等高級用法一網打盡。此外,我還將分享一些獨特的見解和研究發現,希望能給你帶來新的啟發。文章的結尾,我們將有一個 "One More Thing" 環節,我會分享一個很特別但又很少人知... ......

    uj5u.com 2023-05-18 07:38:49 more
  • 實驗一 Java編程環境使用

    ###第1關 String類的常用方法 package step1; public class StringExample { public static void main(String args[]) { /********* Begin *********/ String s1 = new S ......

    uj5u.com 2023-05-18 07:38:46 more
  • 多執行緒基礎知識點梳理

    基礎概念 行程(process):行程是計算機中的一個任務,比如打開瀏覽器、IntelliJ IDEA。 執行緒(thread):行程內部有多個子任務,叫執行緒。比如IDEA在敲代碼的同時還能自動保存、自動導包,都是子執行緒做的。 行程和執行緒的關系就是一個行程包含一個或多個執行緒。 執行緒是作業系統調度的最小 ......

    uj5u.com 2023-05-18 07:38:34 more
  • Java中synchronized的優化

    為了實作高效并發,虛擬機對 synchronized 做的一系列的鎖優化措施。包括:適應性自旋、鎖消除、鎖粗化、輕量級鎖、偏向鎖。 ......

    uj5u.com 2023-05-18 07:37:46 more
  • python中set和frozenset方法和區別

    set(可變集合)與frozenset(不可變集合)的區別: set無序排序且不重復,是可變的,有add(),remove()等方法。既然是可變的,所以它不存在哈希值。基本功能包括關系測驗和消除重復元素. 集合物件還支持union(聯合), intersection(交集), difference( ......

    uj5u.com 2023-05-18 07:37:00 more