Python實現將照片變成卡通圖片的方法【基于opencv】
本文實例講述了Python實現將照片變成卡通圖片的方法。分享給大家供大家參考,具體如下:
之前的文章介紹了使用Photoshop將照片變成卡通圖片,今次介紹用代碼來實現這項任務,可以就此探查各種濾鏡的內部機制。
制作環(huán)境:Windows10,Python2.7,Anaconda
任務描述:將D盤某文件夾中的所有圖片使用代碼進行卡通化,然后保存到另一文件夾中。
如前文所述,卡通化的關鍵是強化邊緣與減少色彩,所以使用Photoshop進行卡通化的時候就使用了照亮邊緣和干筆畫的濾鏡來處理。使用代碼處理圖片的時候也是在對邊緣和色彩上做文章。以下使用OpenCV庫來對照片處理,大致分為四步來完成。
1. 應用雙邊濾波器來減少圖像的色彩
2. 將彩色圖像轉換為灰度,應用中值濾波器減少圖像中的圖像噪點
3. 使用自適應閾值處理灰度圖像創(chuàng)建輪廓
4. 將來自步驟1的彩色圖像與來自步驟3的輪廓疊加
第1步:減少圖像色彩
因為雙邊濾波器平滑平坦區(qū)域同時能保持邊緣清晰,所以很適合于將RGB圖像轉換為卡通。雖然速度好像慢一些一個技巧是重復(例如,通過num_bilateral = 7七次)應用小雙邊濾波器,而不是只用一次大雙邊濾波器。
import cv2 num_down = 2 # 縮減像素采樣的數目 num_bilateral = 7 # 定義雙邊濾波的數目 img_rgb = cv2.imread("img_example.jpg") # 用高斯金字塔降低取樣 img_color = img_rgb for _ in xrange(num_down): img_color = cv2.pyrDown(img_color) # 重復使用小的雙邊濾波代替一個大的濾波 for _ in xrange(num_bilateral): img_color = cv2.bilateralFilter(img_color, d=9, sigmaColor=9, sigmaSpace=7) # 升采樣圖片到原始大小 for _ in xrange(num_down): img_color = cv2.pyrUp(img_color)
cv2.bilateralFilter
中的三個參數控制像素鄰域的直徑(d)和顏色空間中的濾波器的標準偏差(sigmaColor)以及坐標空間(sigmaSpace)。
第2步:轉換為灰度,并使用中值濾波器減少噪點
OpenCV在邊緣檢測方面提供了多種選擇,自適應閾值處理的優(yōu)點是可以檢測圖像的每個小領域中最突出的特征,獨立于圖像的整體屬性。
應用中值濾波器減少圖像的色彩。將原始的彩色圖片轉換為灰度圖片,接著應用中值模糊來減少灰度圖像中的噪聲。
# 轉換為灰度并使其產生中等的模糊 img_gray = cv2.cvtColor(img_rgb, cv2.COLOR_RGB2GRAY) img_blur = cv2.medianBlur(img_gray, 7)
第3步:創(chuàng)建輪廓
在降噪之后,就可以安全地應用自適應閾值來創(chuàng)建輪廓。 即使存在一些圖像噪聲,blockSize = 9的cv2.ADAPTIVE_THRESH_MEAN_C
算法也會確保將閾值應用于9x9鄰域的平均值減去C = 2。
# 檢測到邊緣并且增強其效果 img_edge = cv2.adaptiveThreshold(img_blur, 255, cv2.ADAPTIVE_THRESH_MEAN_C, cv2.THRESH_BINARY, blockSize=9, C=2)
第4步:合并輪廓與彩色圖片
最后一步是將經處理的彩色圖像(img_color)與邊緣掩碼(img_edge)組合。至此,一個可以卡通化圖片的原始代碼就寫好了。圖片效果有點理工的審美,不夠文藝,效果上輸給了Photoshop,但是在效率上扳回一城。
# 轉換回彩色圖像 img_edge = cv2.cvtColor(img_edge, cv2.COLOR_GRAY2RGB) img_cartoon = cv2.bitwise_and(img_color, img_edge) # 顯示圖片 cv2.imshow("cartoon", img_cartoon)
最后將上述代碼封裝成函數。導入python的os模塊來方便文件處理。
# -*- coding: utf-8 -*- import cv2 import os def cartoonise(picture_name): imgInput_FileName = picture_name imgOutput_FileName = "D:\pythonpractice\CartoonImage\cartoon" + picture_name num_down = 2 #縮減像素采樣的數目 num_bilateral = 7 #定義雙邊濾波的數目 img_rgb = cv2.imread(imgInput_FileName) #讀取圖片 #用高斯金字塔降低取樣 img_color = img_rgb for _ in xrange(num_down): img_color = cv2.pyrDown(img_color) #重復使用小的雙邊濾波代替一個大的濾波 for _ in xrange(num_bilateral): img_color = cv2.bilateralFilter(img_color,d=9,sigmaColor=9,sigmaSpace=7) #升采樣圖片到原始大小 for _ in xrange(num_down): img_color = cv2.pyrUp(img_color) #轉換為灰度并且使其產生中等的模糊 img_gray = cv2.cvtColor(img_rgb, cv2.COLOR_RGB2GRAY) img_blur = cv2.medianBlur(img_gray, 7) #檢測到邊緣并且增強其效果 img_edge = cv2.adaptiveThreshold(img_blur,255, cv2.ADAPTIVE_THRESH_MEAN_C, cv2.THRESH_BINARY, blockSize=9, C=2) #轉換回彩色圖像 img_edge = cv2.cvtColor(img_edge, cv2.COLOR_GRAY2RGB) img_cartoon = cv2.bitwise_and(img_color, img_edge) # 保存轉換后的圖片 cv2.imwrite(imgOutput_FileName, img_cartoon) ImageList = [] #建立空的List #循環(huán)讀取"D:\pythonpractice\Image"中的文件名 for filename in os.listdir(r"D:\pythonpractice\Image"): ImageList.append(filename) #將文件名添加到ImageList for i in ImageList: #循環(huán)讀取ImageList中的文件名,將其進行卡通化處理 print("正在卡通化" + i) cartoonise(i)
更多關于Python相關內容感興趣的讀者可查看本站專題:《Python圖片操作技巧總結》、《Python數據結構與算法教程》、《Python Socket編程技巧總結》、《Python函數使用技巧總結》、《Python字符串操作技巧匯總》、《Python入門與進階經典教程》及《Python文件與目錄操作技巧匯總》
希望本文所述對大家Python程序設計有所幫助。
相關文章
Python ''takes exactly 1 argument (2 given)'' Python error
這篇文章主要介紹了Python 'takes exactly 1 argument (2 given)' Python error的相關資料,需要的朋友可以參考下2016-12-12Django Form and ModelForm的區(qū)別與使用
這篇文章主要介紹了Django Form and ModelForm的區(qū)別與使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-12-12