OpenCV圖片漫畫效果的實現(xiàn)示例
我們隨手拍攝的照片,很難達到攝影師的水準,因此不管是手機上還是電腦內(nèi),都有一些軟件可以添加特效讓照片更好看,手機拍攝時也有即時的美化效果。不過我比較好奇漫畫特效,但是一直在網(wǎng)上看到別人的成品而找不到針對性的軟件,因此只有自己實現(xiàn)一下,雖然跟專業(yè)的還有差距,但效果還不錯。
本次使用 OpenCV,采用 Python 實現(xiàn)。
對比現(xiàn)實中的畫畫,一般是先畫出邊緣輪廓使整體規(guī)劃好,再填充顏色使其完整,因此在這里我們也采用這種方式。不過對圖片直接操作與從零開始著筆不一樣,要將原始圖片進行兩次不同的處理,再將處理后的兩個圖片疊加。
邊緣輪廓
漫畫中不管是人物還是風景,刻畫的細節(jié)有限,因此需要把重要以及有特色的部分體現(xiàn)出來,數(shù)量要適當。
輪廓通過四步操作:
import cv2 img = cv2.imread("example.jpg") img_copy = img # 灰度處理 img_gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY) # 平滑操作,去除噪聲 img_blur = cv2.medianBlur(img_gray, 5) # 通過閾值提取輪廓 img_edge = cv2.adaptiveThreshold(img_blur, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY, blockSize=9, C=3) # 將灰度圖片變成 3 通道,用于后續(xù)合并 img_edge = cv2.cvtColor(img_edge, cv2.COLOR_GRAY2BGR)
這里通過二值化的方式將輪廓提取出來,采用自適應閾值二值化函數(shù),基于像素周圍的小區(qū)域確定像素的閾值,可以將有區(qū)別的部分的界限提取出來,恰如漫畫對象中黑色粗體輪廓,且細節(jié)得當。因閾值處理只能針對灰度圖像,因此需要先將彩色圖像轉(zhuǎn)換為單通道的灰度圖像,且為了去除描繪對象內(nèi)部的冗余細節(jié),還要對圖像進行平滑處理,使顏色過度得緩慢一些,畢竟漫畫中顏色的應用沒有現(xiàn)實生活中那么復雜,這樣得出的輪廓就比較好。
看一下效果:
對比原圖:
將兩個重要的方法介紹一下:
中值濾波:cv2.medianBlur(img, ksize)
主要是后面的參數(shù),代表內(nèi)核區(qū)域的邊長,必須是大于1的奇數(shù),如3、5、7……方法提取內(nèi)核區(qū)域下所有像素的中值,并將中心元素替換為該中值
自適應閾值二值化:cv2.adaptiveThreshold(src, maxval, thresh_type, type, Block Size, C)
- src: 輸入圖,只能輸入單通道圖像,通常來說為灰度圖
- dst: 輸出圖
- maxval: 當像素值超過了閾值(或者小于閾值,根據(jù)type來決定),所賦予的值
- thresh_type: 閾值的計算方法,包含以下2種類型:cv2.ADAPTIVE_THRESH_MEAN_C; cv2.ADAPTIVE_THRESH_GAUSSIAN_C.
- type:二值化操作的類型,與固定閾值函數(shù)相同,包含以下5種類型: cv2.THRESH_BINARY; cv2.THRESH_BINARY_INV; cv2.THRESH_TRUNC; cv2.THRESH_TOZERO;cv2.THRESH_TOZERO_INV.
- Block Size: 圖片中分塊的大小
- C :閾值計算方法中的常數(shù)項
顏色填充
邊緣輪廓已經(jīng)描繪好了,再添加顏料后就完整了。這里就比較簡單了,只需要將原圖片的顏色細致度降低些就行了。
代碼如下:
for _ in range(2) # 降低分辨率 img_copy = cv2.pyrDown(img_copy) for _ in range(5): # 圖像平滑,保留邊緣 img_copy = cv2.bilateralFilter(img_copy, d=9, sigmaColor=9, sigmaSpace=7) img_copy = cv2.resize(img_copy, (img.shape[1], img.shape[0]), interpolation=cv2.INTER_CUBIC)
顏色要比較平滑,不能像現(xiàn)實生活中這么細致,先采用圖像金字塔將分辨率降低,并采用雙邊濾波去除噪聲,可以平滑平面區(qū)域,同時保持邊緣清晰。分辨率降低后圖像會變小,因此最后要將圖像放大為原來的大小,雖然圖像金字塔有專門的方法可以將圖像放大,但是尺寸可能會有一兩點變化,合并過程中要兩個圖像完全一樣大,所以這里直接按尺寸放大。
看一下效果:
將兩個重要的方法介紹一下:
分辨率降低:cv2.pyrDown(src, dst=None, dstsize=None, borderType=None)
一般只需要輸入圖像就行了,它會直接將圖像長寬減半
雙邊濾波:cv2.bilateralFilter(src, d, sigmaColor, sigmaSpace[, dst[, borderType]])
- src:輸入圖像
- d:過濾時周圍每個像素領(lǐng)域的直徑
- sigmaColor:在color space中過濾sigma。參數(shù)越大,臨近像素將會在越遠的地方mix。
- sigmaSpace:在coordinate space中過濾sigma。參數(shù)越大,那些顏色足夠相近的的顏色的影響越大。
合并
img_cartoon = cv2.bitwise_and(img_copy, img_edge) cv2.imshow("cartoon", img_cartoon) cv2.waitKey(0) cv2.destroyAllWindows()
cv2.bitwise_and()
是對二進制數(shù)據(jù)進行“與”操作,即對圖像(灰度圖像或彩色圖像均可)每個像素值進行二進制“與”操作:1&1=1,1&0=0,0&1=0,0&0=0
最后結(jié)果:
到此這篇關(guān)于OpenCV圖片漫畫效果的實現(xiàn)示例的文章就介紹到這了,更多相關(guān)OpenCV 圖片漫畫內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺談Python中進程的創(chuàng)建與結(jié)束
這篇文章主要介紹了淺談Python中進程的創(chuàng)建與結(jié)束,但凡是硬件,都需要有操作系統(tǒng)去管理,只要有操作系統(tǒng),就有進程的概念,就需要有創(chuàng)建進程的方式,需要的朋友可以參考下2023-07-07python django框架中使用FastDFS分布式文件系統(tǒng)的安裝方法
這篇文章主要介紹了python-django框架中使用FastDFS分布式文件系統(tǒng)的安裝方法,本文圖文并茂給大家介紹的非常詳細,具有一定的參考借鑒價值 ,需要的朋友可以參考下2019-06-06使用Python實現(xiàn)將數(shù)據(jù)寫入Excel工作表
在數(shù)據(jù)處理和報告生成等工作中,Excel?表格是一種常見且廣泛使用的工具,本文中將介紹如何使用?Python?寫入數(shù)據(jù)到?Excel?表格,并提供更高效和準確的?Excel?表格數(shù)據(jù)寫入方案,需要的可以參考下2024-01-01關(guān)于python中remove的一些坑小結(jié)
這篇文章主要給大家介紹了關(guān)于python中remove的一些坑,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-01-01