news 2026/6/25 17:02:55

Python批量图片拼接脚本:支持行列布局、最后一行居中、自然排序

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Python批量图片拼接脚本:支持行列布局、最后一行居中、自然排序

平时写论文、做报告或者整理素材时,经常需要把多张图片拼成一张大图。试过不少在线工具和本地软件,要么是没法批量处理,要么是最后一行图片没填满时就左对齐,空出一大块特别丑。

索性自己写了个Python脚本,用Pillow库实现,核心解决了几个痛点:

  • 最后一行自动居中:没填满的行不会傻呵呵左对齐,视觉上更平衡
  • 自然排序:确保1.jpg2.jpg10.jpg按数字顺序排,不会出现1, 10, 2的尴尬
  • 保留原图质量:支持DPI、色彩模式继承,JPG会设最高质量
  • 灵活配置:行数、列数、每组几张图、底部留白都能改

关键代码摘出与解释

1. 自然排序:解决10.jpg排在2.jpg前面的问题

系统默认的字符串排序是按字符ASCII码比较的,'10'的第一个字符是'1',所以会排在'2'前面。这个函数通过正则把文件名拆分成“数字段”和“非数字段”,把数字段转成整数后再比较,符合人类的阅读习惯。

def natural_sort_key(s): """自然排序:确保 1.jpg, 2.jpg, 10.jpg 按数字顺序排列""" return [int(text) if text.isdigit() else text.lower() for text in re.split('([0-9]+)', s)]

使用场景:在读取文件列表后,通过sorted(..., key=natural_sort_key)调用即可。


2. 核心逻辑:最后一行图片居中计算

这是脚本最实用的部分。思路是:先按“列数”把图片切分成行,然后判断当前行是否填满。如果没填满,就算出两边需要留多少空白,把图片“挤”到中间。

# ⭐ 核心:按行切分 rows = [] for i in range(0, len(group_images), COLS): rows.append(group_images[i:i + COLS]) for row_idx, row_images in enumerate(rows): y = row_idx * img_height num_in_row = len(row_images) # ⭐ 核心:最后一行居中 if num_in_row == COLS: offset_x = 0 # 满行,左对齐 else: # 不满行,计算居中偏移量 total_width = num_in_row * img_width offset_x = (canvas_width - total_width) // 2

逻辑说明

  • offset_x是这一行第一张图的起始X坐标。
  • 如果满行,offset_x为0,从最左边开始贴。
  • 如果不满行,用(画布宽 - 当前行图片总宽) // 2算出左边留白,实现居中。

3. 图片一致性处理:尺寸、模式统一

为了防止拼图出现错位或色差,需要以第一张图为基准,统一所有图片的尺寸和色彩模式。

with Image.open(img_path) as img: # 尺寸统一:如果不一致,用LANCZOS算法高质量重采样 if img.size != (img_width, img_height): print(f"警告: {os.path.basename(img_path)} 尺寸不一致,已调整") img = img.resize((img_width, img_height), Image.Resampling.LANCZOS) # 模式统一:比如有的是RGB,有的是RGBA,统一成基准图的模式 if img.mode != img_mode: img = img.convert(img_mode) canvas.paste(img, (x, y))

注意Image.Resampling.LANCZOS是Pillow 9.1.0之后的写法,如果用的是旧版本,可能需要改成Image.LANCZOS


4. 高质量保存:保留DPI与JPG画质

如果是用于论文打印,保留DPI很重要;JPG格式默认压缩会损失画质,这里强制设为最高质量。

save_params = {} # 如果是JPG/JPEG,设置最高质量,禁用色度子采样 if input_extension.lower() in ['.jpg', '.jpeg']: save_params['quality'] = 100 save_params['subsampling'] = 0 # 如果原图有DPI信息,继承下来 if img_dpi: save_params['dpi'] = img_dpi canvas.save(save_path, **save_params)

完整代码

直接复制下面的代码,保存为image_collage.py即可使用。使用前请务必修改手动配置区的路径。

import re from PIL import Image import os # ==================== 手动配置区 ==================== TARGET_PATH = r'你的图片文件夹路径' # 请修改此处 IMAGES_PER_GROUP = 3 # 每组几张图 ROWS = 2 # 行数 COLS = 2 # 列数 BOTTOM_PADDING = 0 # 底部留白(像素) # =================================================== def natural_sort_key(s): """自然排序:确保 1.jpg, 2.jpg, 10.jpg 按数字顺序排列""" return [int(text) if text.isdigit() else text.lower() for text in re.split('([0-9]+)', s)] def create_collage(folder_path): # 1. 读取并筛选图片 valid_extensions = ('.jpg', '.jpeg', '.png', '.bmp', '.webp', '.tiff') raw_files = [ f for f in os.listdir(folder_path) if f.lower().endswith(valid_extensions) ] # 排除之前生成的拼图,防止重复处理 raw_files = [f for f in raw_files if "collage_final" not in f] # 按自然顺序排序并拼接完整路径 image_files = sorted( [os.path.join(folder_path, f) for f in raw_files], key=natural_sort_key ) if not image_files: print("未找到图片,请检查路径。") return # 2. 读取第一张图作为基准,获取尺寸、模式、DPI等信息 with Image.open(image_files[0]) as first_img: img_width, img_height = first_img.size input_extension = os.path.splitext(image_files[0])[1] img_mode = first_img.mode img_dpi = first_img.info.get('dpi') print(f"\n--- 原始分辨率识别 ---") print(f"单图尺寸: {img_width} x {img_height}") print(f"色彩模式: {img_mode}") print(f"输出格式: {input_extension}") if img_dpi: print(f"DPI: {img_dpi}") # 3. 计算画布总尺寸 canvas_width = COLS * img_width canvas_height = ROWS * img_height + BOTTOM_PADDING # 4. 分组(如果图片很多,可以分成多张拼图) groups = [ image_files[i:i + IMAGES_PER_GROUP] for i in range(0, len(image_files), IMAGES_PER_GROUP) ] for group_num, group_images in enumerate(groups): # 创建白色背景画布 canvas = Image.new(img_mode, (canvas_width, canvas_height), color='white') # ⭐ 核心逻辑:将当前组的图片按行切分 rows = [] for i in range(0, len(group_images), COLS): rows.append(group_images[i:i + COLS]) # 逐行粘贴图片 for row_idx, row_images in enumerate(rows): y = row_idx * img_height num_in_row = len(row_images) # ⭐ 核心逻辑:计算当前行的X偏移量(实现居中) if num_in_row == COLS: offset_x = 0 else: total_width = num_in_row * img_width offset_x = (canvas_width - total_width) // 2 # 逐列粘贴图片 for col_idx, img_path in enumerate(row_images): x = offset_x + col_idx * img_width try: with Image.open(img_path) as img: # 统一尺寸 if img.size != (img_width, img_height): print(f"警告: {os.path.basename(img_path)} 尺寸不一致,已调整") img = img.resize((img_width, img_height), Image.Resampling.LANCZOS) # 统一色彩模式 if img.mode != img_mode: img = img.convert(img_mode) canvas.paste(img, (x, y)) except Exception as e: print(f"处理出错: {img_path}, 原因: {e}") # 5. 保存文件 suffix = f"_{group_num + 1}" if len(groups) > 1 else "" save_name = f"collage_final{suffix}{input_extension}" save_path = os.path.join(folder_path, save_name) save_params = {} # JPG特殊处理:最高质量
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/25 17:02:28

使用 Java 提取 HTML 文件中的纯文本内容

、实现原理Free Spire.Doc for Java 是一款免费库,其核心设计围绕 Word 文档的段落、节、表格等元素展开。当调用 loadFromFile 方法并指定 FileFormat.Html 时,库内部会将 HTML 标签、样式和文本映射到自己的文档对象模型中。随后调用 getText() 方法&a…

作者头像 李华
网站建设 2026/6/25 16:55:19

MuleSoft+LangChain企业级AI编排实战:打通数据与大模型的数字脐带

1. 项目概述:当企业级集成遇上大模型,为什么“拼积木”式AI落地正在失效?我在金融行业做系统集成顾问整整十二年,从最早的SOAP WebService手写WSDL文档,到后来用MuleSoft搭API网关,再到去年开始被客户拉着一…

作者头像 李华
网站建设 2026/6/25 16:54:56

3分钟解锁B站缓存视频:m4s-converter无损转换工具全攻略

3分钟解锁B站缓存视频:m4s-converter无损转换工具全攻略 【免费下载链接】m4s-converter 一个跨平台小工具,将bilibili缓存的m4s格式音视频文件合并成mp4 项目地址: https://gitcode.com/gh_mirrors/m4/m4s-converter 还在为B站缓存视频只能在官方…

作者头像 李华
网站建设 2026/6/25 16:53:11

连锁拓店 / 公装避坑指南②:预算坑

很多连锁品牌拓店翻车,不是设计不好、施工太差,而是栽在了预算报价上。很多加盟商、品牌负责人,只对比总价选低价,看不懂报价套路,开工后被动加价,单店成本直接失控,批量开店更是亏得连片亏损。…

作者头像 李华
网站建设 2026/6/25 16:48:10

AI赋能红队自动化:HexStrike平台10分钟复现Citrix漏洞攻防解析

1. 项目概述:当红队工具遇上AI,攻防演练进入“快进”时代最近在安全圈里,一个名为HexStrike的工具讨论度很高。它被定位为一款“AI赋能的红队自动化平台”,简单来说,就是让安全测试人员(红队)能…

作者头像 李华
网站建设 2026/6/25 16:47:55

力扣138随机链表的复制

给你一个长度为 n 的链表,每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。 构造这个链表的 深拷贝。 深拷贝应该正好由 n 个 全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的 ne…

作者头像 李华