需求說明:制作一個馬賽克圖片,將幾萬張圖片進行合成,每個馬賽克格子的尺寸為15x15,

完整原始碼在文末有說明,需要的可以去下載哈,
【閱讀全文】 看一下生成的馬賽克圖片的效果:

需要使用到的python模塊包如下:
import cv2 # pip install opencv-python # 影像處理庫
import glob # 匯入檔案處理庫
import argparse # 命令列決議庫
import numpy as np # 資料處理庫
from tqdm import tqdm # 匯入進度條庫
from itertools import product # 匯入迭代器庫
import logging # 匯入日志庫
設定logging模塊日志列印,
logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(name)s - %(levelname)s - %(message)s')
logger = logging.getLogger("圖片合成器")
logger.setLevel(logging.DEBUG)
撰寫函式read_source_images()提取符合影像顏色要求的圖片物件并計算平均值,
def read_source_images(source_images_path, block_size):
'''
提取符合影像顏色要求的圖片物件并計算平均值
:param source_images_path: 源圖片路徑
:param block_size: 每個圖片的尺寸
:return: 符合要求的圖片物件陣列、顏色平均值陣列
'''
logger.info("開始源圖片篩選及顏色平均值計算處理")
source_images = [] # 初始化源影像串列
avg_colors = [] # 平均顏色串列
'''使用進度條進行遍歷源圖片的檔案夾'''
for image_path in tqdm(glob.glob("{}/*.jpg".format(source_images_path))):
try: # 此處加入例外處理,若出現處理例外的圖片則跳過該圖片處理
# 讀取彩色圖片
img_obj = cv2.imread(image_path, cv2.IMREAD_COLOR)
'''
img_obj.shape[-1]讀取影像的通道數,通道值為3表示每個像素點的顏色取值范圍(0-255,0-255,0-255),
將通道值不等于3的圖片跳過,
'''
if img_obj.shape[-1] != 3:
continue
# 重新設定圖片的尺寸
img_obj = cv2.resize(img_obj, (block_size, block_size))
# 計算該影像顏色的平均值
avg_color = np.sum(np.sum(img_obj, axis=0), axis=0) / (block_size * block_size)
# 將符合要求的影像物件添加到陣列中
source_images.append(img_obj)
# 將符合要求的影像顏色平均值添加到陣列中
avg_colors.append(avg_color)
except:
logger.error("例外圖片路徑:" + image_path)
logger.info("結束源圖片篩選及顏色平均值計算處理")
return source_images, np.array(avg_colors)
撰寫parse_args()函式,用于決議檔案相關的引數,之后需要獲取引數時直接從引數決議器中提取使用即可,
def parse_args():
'''
引數決議函式
:return:
'''
logger.info("開始檔案引數決議處理")
parser = argparse.ArgumentParser('圖片檔案引數決議器')
# 添加目標影像路徑
parser.add_argument('--targetpath', type=str, default='target.jpg', help='目標影像路徑')
# 添加輸出影像路徑
parser.add_argument('--outputpath', type=str, default='output.jpg', help='輸出影像的路徑')
# 源圖片檔案路徑
parser.add_argument('--sourcepath', type=str, default='source_images', help='源圖片檔案夾路徑')
# 需要轉換的每個圖片的目標尺寸
parser.add_argument('--blocksize', type=int, default=15, help='每個圖片的目標尺寸')
# 決議引數并回傳
args = parser.parse_args()
logger.info("結束檔案引數決議處理")
return args
撰寫main_merage()函式,用于實作馬賽克圖片的正式合成,
def main_merage(params):
'''
圖片合成處理函式
:param params: 檔案引數
:return:
'''
# 獲取目標圖片物件,默認按彩色方式讀取
target_image_obj = cv2.imread(params.targetpath)
# 根據目標圖片物件,生成對應的零矩陣
output_image_obj = np.zeros(target_image_obj.shape, np.uint8)
# 獲取符合要求的源圖片陣列與平均顏色陣列
source_images, avg_colors = read_source_images(params.sourcepath, params.blocksize)
# 根據目標圖片的長、寬執行遍歷
'''target_image_obj.shape[1]、target_image_obj.shape[0]獲得圖片的長、寬'''
logger.info("開始圖片合成處理")
for i, j in tqdm(product(range(int(target_image_obj.shape[1] / params.blocksize)),
range(int(target_image_obj.shape[0] / params.blocksize)))):
block = target_image_obj[j * params.blocksize: (j + 1) * params.blocksize,
i * params.blocksize: (i + 1) * params.blocksize, :]
avg_color = np.sum(np.sum(block, axis=0), axis=0) / (params.blocksize * params.blocksize)
distances = np.linalg.norm(avg_color - avg_colors, axis=1)
idx = np.argmin(distances)
output_image_obj[j * params.blocksize: (j + 1) * params.blocksize,
i * params.blocksize: (i + 1) * params.blocksize, :] = \
source_images[idx]
cv2.imwrite(params.outputpath, output_image_obj)
cv2.imshow('輸出生成的圖片', output_image_obj)
logger.info("結束圖片合成處理")
使用前面文章中提到的百度圖片下載器下載我們需要的源圖片,

沒有下載的到公眾號回復"百度圖片下載器"去下載就可以了,
源圖片準備的越多越好,我這里直接準備了兩萬張美女圖片作為源圖片,如果想讓生成的圖片更加逼真就下載更多的源圖片,


【往期精彩】
小工具批量將mp3音頻格式轉換為wav格式
不用H5,直接使用pywebio模塊實作網頁
python回呼函式能做什么?
解決pyinstaller打包程序中外部資源無法加載的問題 ...
pyqt5做了一個二維碼生成器,已打包成exe可執行程式...
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/404369.html
標籤:python
上一篇:Python再次拿第一,我一點都不意外,只是有些事情并非人人都看得清楚
下一篇:軟體測驗的底層邏輯是什么?
