多线程带进度条的python批量压缩图片
创始人
2024-01-17 15:56:10
0
from PIL import Image
import os
import logging
from datetime import datetime
from concurrent.futures import ThreadPoolExecutor
from tqdm import tqdm

def resize_and_compress_image(input_path, quality, size_threshold_kb, max_width):
    try:
        # 获取图片文件大小
        file_size_kb = os.path.getsize(input_path) / 1024

        # 仅当图片大小大于300KB时进行压缩
        if file_size_kb > size_threshold_kb:
            # 打开图片
            img = Image.open(input_path)

            # 调整宽度为最大值1000像素
            if img.width > max_width:
                ratio = max_width / float(img.width)
                new_height = int(float(img.height) * float(ratio))
                img = img.resize((max_width, new_height), Image.LANCZOS)

            # 覆盖保存为JPEG或PNG格式,根据文件类型设置相应的参数
            if input_path.lower().endswith(('.jpg', '.jpeg')):
                img.save(input_path, 'JPEG', quality=quality, optimize=True)
            elif input_path.lower().endswith('.png'):
                img.save(input_path, 'PNG', optimize=True)

            logging.info(f"成功压缩图片: {input_path}")
            return True
        else:
            logging.info(f"跳过小于{size_threshold_kb}KB的图片: {input_path}")
            return False
    except Exception as e:
        logging.error(f"压缩图片时发生错误: {input_path}. 错误信息: {str(e)}")
        return False

def resize_and_compress_images(input_folder, quality=85, size_threshold_kb=300, max_width=1000, num_threads=4):
    # 设置日志
    log_file = f"compression_log_{datetime.now().strftime('%Y%m%d%H%M%S')}.txt"
    logging.basicConfig(filename=log_file, level=logging.DEBUG, format='%(asctime)s - %(levelname)s - %(message)s')

    # 获取总图片数
    total_images = sum([len(files) for _, _, files in os.walk(input_folder)])

    # 初始化tqdm
    progress_bar = tqdm(total=total_images, desc="处理进度")

    # 统计信息
    compressed_images = 0
    total_original_size = 0

    # 设置线程池
    with ThreadPoolExecutor(max_workers=num_threads) as executor:
        futures = []

        for root, dirs, files in os.walk(input_folder):
            for image_file in files:
                input_path = os.path.join(root, image_file)

                # 获取图片文件大小
                file_size_kb = os.path.getsize(input_path) / 1024
                total_original_size += file_size_kb

                # 提交任务到线程池
                future = executor.submit(resize_and_compress_image, input_path, quality, size_threshold_kb, max_width)
                futures.append(future)

        # 等待所有任务完成
        for future in tqdm(futures, total=len(futures), desc="多线程处理"):
            if future.result():
                compressed_images += 1

            # 更新进度条
            progress_bar.update(1)

    # 关闭进度条
    progress_bar.close()

    # 记录统计信息
    logging.info(f"总共图片数: {total_images}")
    logging.info(f"成功压缩图片数: {compressed_images}")

    # 输出日志文件路径
    print(f"日志文件路径: {os.path.abspath(log_file)}")

if __name__ == "__main__":
    # 输入文件夹(包含待压缩的JPEG和PNG图片)
    input_folder = "input_images"

    # 设置压缩质量(仅适用于JPEG,0-100,100为最高质量)
    compression_quality = 85

    # 设置图片大小阈值(大于该阈值才进行压缩)
    size_threshold_kb = 300

    # 设置最大宽度
    max_width = 1000

    # 设置线程数
    num_threads = 4

    # 执行图片压缩(直接覆盖原文件)
    resize_and_compress_images(input_folder, quality=compression_quality, size_threshold_kb=size_threshold_kb, max_width=max_width, num_threads=num_threads)

    print("图片压缩完成。日志文件详见压缩日志。")

相关内容

热门资讯

Windows 11 和 10... Windows 11/10 文件夹属性中缺少共享选项卡 – 已修复 1.检查共享选项卡是否可用 右键...
Radmin VPN Wind... Radmin VPN 是一款免费且用户友好的软件,旨在牢固地连接计算机以创建一个有凝聚力的虚拟专用网...
如何修复 Steam 内容文件... Steam 内容文件锁定是当您的 Steam 文件无法自行更新时出现的错误。解决此问题的最有效方法之...
在 Windows 11 中打... 什么是链路状态电源管理? 您可以在系统控制面板的电源选项中看到链接状态电源管理。它是 PCI Exp...
事件 ID 7034:如何通过... 点击进入:ChatGPT工具插件导航大全 服务控制管理器 (SCM) 负责管理系统上运行的服务的活动...
Hive OS LOLMine... 目前不清退的交易所推荐: 1、全球第二大交易所OKX欧意 国区邀请链接: https://www.m...
在 iCloud 上关闭“查找... 如果您是 Apple 的长期用户,您肯定会遇到过 Find My 应用程序,它本机安装在 iPhon...
iPhone 屏幕上有亮绿色斑... iPhone 是市场上最稳定的智能手机之一,这主要归功于专为它们设计的 iOS 操作系统。然而,他们...
balenaEtcher烧录后... balenaEtcher烧录后u盘或者内存卡无法识别不能使用的解决方法想要恢复原来的方法,使用win...
farols1.1.501.0... faro ls 1.1.501.0(64bit)可以卸载,是一款无需连接外部PC机或笔记本计算机即可...