基于Python实现千图成像工具的示例代码

 更新时间:2022年07月26日 16:03:00   作者:锋小刀  
千图成像也就是用N张图片组成一张图片的效果。这篇文章将利用Python语言编写一个简单的千图成像工具,感兴趣的小伙伴可以了解一下

千图成像也就是用N张图片组成一张图片的效果。制作方法有很多的,最常见的如用ps、懒人图云、foto-mosaik-edda这些制作。

千图成像的效果我大致分为两类:一为直接用N张图片根据底图的像素颜色、大小,一张张的组成底图,如foto-mosaik-edda;二为用N张图片根据底图的像素大小,组成一张与底图大小相仿的图片,再把二者合成,经调整透明度而成的图片,如ps。

第一种算是真正意义的千图成像,但如果选的图片不够底图的像素颜色匹配,就会造成生成的图片畸形,但如果选择的图片够好,最终的效果会非常好;第二种的效果就比较平淡了,但对选择的图片没什么要求,生成的图片比较正常。

二者的效果各有千秋,而本文使用python实现的是第二种方法,最后制成GUI。

前置

本文使用PySimpleGUI进行GUI设计,PIL、numpy、random 进行图片处理,os进行文件操作:

import PySimpleGUI as sg
from PIL import Image
import os
import numpy as np
import random

相关库使用pip命令安装即可:

pip install 库名

GUI制作

为了以后方便使用,不用一次次跑程序,而在原有的程序基础上进行GUI制作,最后打包成.exe可执行文件。

GUI界面设计

对于GUI界面的功能只需要设定五个功能即可:

  • 选择底图功能
  • 选择组图功能
  • 事件展示区域
  • 启动工具按钮
  • 退出工具按钮

最终设计代码如下:

# 主题设置
sg.theme('LightBrown3')

# 布局设置
layout = [
    [sg.Frame(layout=[
        [sg.InputText(key='image_file', size=(32, 1), font=("微软雅黑", 10), enable_events=True),
        # 设定能选择的图片格式
         sg.FileBrowse('选择底图',
                       file_types=(("Text Files", "*.png*"), ("Text Files", "*.jpg*"), ("Text Files", "*.jpeg*")),
                       font=("微软雅黑", 12)),

         sg.Button('选择组图', font=("微软雅黑", 12)),
         ],
    ],
        title='内容选择', title_color='blue', font=("微软雅黑", 10), relief=sg.RELIEF_SUNKEN, )],
    [sg.Frame(layout=[
        [sg.Output(size=(51, 10), font=("微软雅黑", 10))],
    ],
        title='信息展示', title_color='blue', font=("微软雅黑", 10), relief=sg.RELIEF_SUNKEN, )],

    [sg.Button('开始生成', font=("微软雅黑", 12)),
     sg.Text('', font=("微软雅黑", 12), size=(27, 0)), sg.Button('退出程序', font=("微软雅黑", 12), button_color='red')]
]
# 创建窗口
window = sg.Window('千图成像', layout, font=("微软雅黑", 12), default_element_size=(80, 1))
while True:
    # 退出按钮
    event, values = window.read()
    if event in (None, '退出程序'):
        break
window.close()

界面效果如下:

GUI界面效果

逻辑设计

获取图片时,因为可以输入路径,可能会造成保存,所以这里加个判断;最后把得到的图片路径存入列表中。

if event == 'image_file':
    files = values['image_file']
    if os.path.exists(files):
        img_Main_file.append(files)
    else:
        print('图片不存在,请重新选择图片!')
        # 弹窗
        sg.popup('图片不存在,请重新选择图片!')

获取组图所在的文件夹路径,依然把得到的路径存入列表中:

if event == '选择组图':
    files = sg.popup_get_folder('请选择选择组图路径:')
    if os.path.exists(files):
        img_secondary_file.append(files)
    else:
        print('文件不存在,请重新选择文件')
        sg.popup('文件不存在,请重新选择文件')

启动按钮,点击时把两个列表传入图片处理函数中:

if event == '开始生成':
    if len(img_Main_file) and len(img_secondary_file) != 0:
        img_save(img_Main_file, img_secondary_file)
    else:
        sg.popup('未选择!')

图片处理

因为无法保证所有图片的大小都一样,所以需要经过一定的处理。图片处理使用的库是PIL和numpy。

修改底图大小

对于底图,我们可以称之为‘容器’,底图的大小决定其组成图片的多少,也可以决定组成图片的像素大小、是否清晰。取出底图的高宽越多,图片越大,图片越清晰;取出底图百分之十的大小,这个数值可以增大,但最好不要超过百分之三十。

open_img = Image.open('./底图.jpg')
# 获取图片本身宽度、高度
width, height = open_img.size
# 重新计算底图高宽,加大底图的像素。取出底图的10%的高宽,用int进行取整
Increase_width = int(width * 0.10) * int(height * 0.10)
Increase_height = ((Increase_width / width) * height // round(height * 0.10)) * round(height * 0.10)
# 更改为重新计算的大小
open_img = open_img.resize((int(Increase_width), int(Increase_height)), Image.ANTIALIAS)

修改组图大小

把组成图片的大小修改为底图的百分之十的大小,这个数值也可以增大:

# 读取文件路径下的图片,并修改大小
img_matrix = []
for e in os.listdir('./image'):
    # 防止文件夹中出现并图片格式的文件
    try:
        img_matrix.append(np.array(Image.open(os.path.join(str(img_files_list[0]), e)).convert('RGB').resize(
            (int(width * 0.10), int(height * 0.10)), Image.ANTIALIAS)))
    except OSError as e:
        print(e)

计算图片填充次数

上面说过,底图的大小决定了组成图片的多少,而下面的代码就是根据底图的大小以及组成图片的大小计算出主图能填充多少图片:

# 计算主图高宽能填充多少图片
width_picture_Fill_frequency = int(Increase_width / int(width * 0.10))
height_picture_Fill_frequency = int(Increase_height / int(height * 0.10))

组图合成

根据底图高宽的10%以及图片填充次数,得出矩阵,然后把组图随机填充到矩阵中:

array_img = np.zeros_like(np.array(open_img))
for i in range(width_picture_Fill_frequency):
    for x in range(height_picture_Fill_frequency):
        array_img[x * int(height * 0.10):(x + 1) * int(height * 0.10),
        i * int(width * 0.10):(i + 1) * int(width * 0.10), :] = random.choice(img_matrix)
array_img = Image.fromarray(array_img)

生成的图片清晰度还是很高的,不过在手机上看比较模糊:

组图效果

图片合成

把底图和组图进行合并,alpha可以调整二者的透明度,最佳为0.7、0.8、0.9。

img = Image.blend(array_img, open_img, alpha=0.8)  # 0.7,0.8,0.9
img.save('千图成像.jpg')

图片效果

GUI打包

打包可以直接使用pyinstaller进行安装;如果你不知道怎么打包,或者不熟悉命令行操作,可以使用前面文章:打包工具,这款打包工具可以简单的满足打包需求。

使用pyinstaller库打包,启动命令行窗口,在命令行窗口cd到文件所在的文件目录中,最后用下面命令进行打包:

pyinstaller -F -w 名称.py

打包时可能会报错:

报错示例

报错源于一个hook-sqlalchemy.py文件,一个简单的解决方法是找到它直接回收删除它(最后暂未发现删除它对打包后的exe文件有什么影响),等打包完成后在放回去即可:

打包过程没出现什么状况,会得到几个文件,进入dist文件夹,就可以看见.exe文件了。

至此,我们就成功利用Python实现了制作千图成像工具。

到此这篇关于基于Python实现千图成像工具的示例代码的文章就介绍到这了,更多相关Python千图成像内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • python中求两个向量的夹角方式

    python中求两个向量的夹角方式

    这篇文章主要介绍了python中求两个向量的夹角方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • Django models文件模型变更错误解决

    Django models文件模型变更错误解决

    这篇文章主要介绍了Django models文件模型变更错误解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-05-05
  • pycharm python代码调试跳出for循环问题

    pycharm python代码调试跳出for循环问题

    这篇文章主要介绍了pycharm python代码调试跳出for循环问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • Pygame的程序开始示例代码

    Pygame的程序开始示例代码

    这篇文章主要介绍了Pygame的程序开始的示例代码,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • Selenium爬取b站主播头像并以昵称命名保存到本地

    Selenium爬取b站主播头像并以昵称命名保存到本地

    这篇文章主要介绍了使用Selenium自动化爬取b站主播头像并以昵称命名保存到本地的方法,代码简单完整,对于大家练习Selenium自动化有一定的帮助,需要的朋友可以参考下
    2021-04-04
  • Pycharm配置远程调试的方法步骤

    Pycharm配置远程调试的方法步骤

    这篇文章主要介绍了Pycharm配置远程调试的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • python爬虫获取小区经纬度以及结构化地址

    python爬虫获取小区经纬度以及结构化地址

    这篇文章主要为大家详细介绍了python爬虫获取小区经纬度,以及结构化的地址,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • Python常用模块用法分析

    Python常用模块用法分析

    这篇文章主要介绍了Python常用模块用法分析,主要包括内置函数、文件操作、正则匹配等等,需要的朋友可以参考下
    2014-09-09
  • python好玩的项目—色情图片识别代码分享

    python好玩的项目—色情图片识别代码分享

    这篇文章主要介绍了python好玩的项目—色情图片识别,具有一定参考价值,需要的朋友可以了解下。
    2017-11-11
  • Django中模板的继承及引用实现

    Django中模板的继承及引用实现

    Django模版引擎中最强大也是最复杂的部分就是模版继承了,页面的代码很多,随随便便就是几百行代码,但是每个页面之中都有相同的元素。本文主要介绍了Django模板的继承及引用,感兴趣的可以了解一下
    2021-08-08

最新评论