欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python調(diào)用PIL庫實現(xiàn)圖片格式轉(zhuǎn)換工具

 更新時間:2025年04月25日 10:37:25   作者:學(xué)習(xí)&實踐愛好者  
這篇文章主要為大家詳細介紹了Python如何調(diào)用PIL庫實現(xiàn)圖片格式轉(zhuǎn)換工具,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

要運行該程序,需要使用第三方庫PIL(pillow),詳情可見Python的圖像處理庫Pillow安裝與使用教程

格式支持:

支持常見圖片格式轉(zhuǎn)換(JPEG, PNG, BMP, GIF, TIFF, WEBP)

轉(zhuǎn)換設(shè)置:

  • 可調(diào)整輸出圖片質(zhì)量(1-100)
  • 支持尺寸調(diào)整(可保持寬高比,保持寬高比時,只需輸入寬度或高度中的一個即可自動計算另一個維度)
  • 可選是否保留元數(shù)據(jù)(EXIF信息)

輸出選項:

  • 自定義輸出路徑
  • 自動生成新文件名(保留原文件名)

使用方法:

(1)通過"添加文件"或"添加文件夾"按鈕選擇圖片

(2)設(shè)置輸出格式和保存路徑(若不指定輸出路徑默認原文件路徑)

(3)根據(jù)需要調(diào)整圖片質(zhì)量和尺寸

(4)點擊"開始轉(zhuǎn)換"按鈕進行轉(zhuǎn)換

若輸出路徑有同名文件提示提供三個選項:

是:覆蓋當(dāng)前文件

否:跳過當(dāng)前文件

 取消:中止整個轉(zhuǎn)換過程

注意事項:

  • 轉(zhuǎn)換透明背景圖片到不支持透明的格式(如JPEG)時,背景會被自動填充為白色
  • 質(zhì)量設(shè)置對不同的格式效果不同(JPEG使用標準質(zhì)量參數(shù),PNG使用壓縮級別)
  • 保持寬高比時,只需輸入寬度或高度中的一個即可自動計算另一個維度

運行界面如下:

源碼如下:

import os
import tkinter as tk
from tkinter import ttk, filedialog, messagebox
from PIL import Image
from pathlib import Path
 
SUPPORTED_FORMATS = ["JPEG", "PNG", "BMP", "GIF", "TIFF", "WEBP"]
 
class ImageConverterApp:
    def __init__(self, root):
        self.root = root
        self.root.title("圖片格式轉(zhuǎn)換工具")
        self.root.geometry("800x600")
 
        # 初始化變量
        self.selected_files = []
        self.output_format = tk.StringVar(value="JPEG")
        self.output_path = tk.StringVar()
        self.quality = tk.IntVar(value=85)
        self.resize_enabled = tk.BooleanVar(value=False)
        self.new_width = tk.IntVar(value=0)
        self.new_height = tk.IntVar(value=0)
        self.keep_ratio = tk.BooleanVar(value=True)
        self.keep_metadata = tk.BooleanVar(value=False)
 
        self.create_widgets()
 
    def create_widgets(self):
        # 文件選擇區(qū)域
        file_frame = ttk.LabelFrame(self.root, text="選擇圖片")
        file_frame.pack(padx=10, pady=5, fill="x")
 
        ttk.Button(file_frame, text="添加文件", command=self.add_files).pack(side="left", padx=5)
        ttk.Button(file_frame, text="添加文件夾", command=self.add_folder).pack(side="left", padx=5)
        ttk.Button(file_frame, text="清空列表", command=self.clear_files).pack(side="right", padx=5)
 
        # 文件列表
        self.file_list = tk.Listbox(self.root, selectmode=tk.EXTENDED)
        self.file_list.pack(padx=10, pady=5, fill="both", expand=True)
 
        # 設(shè)置區(qū)域
        settings_frame = ttk.LabelFrame(self.root, text="轉(zhuǎn)換設(shè)置")
        settings_frame.pack(padx=10, pady=5, fill="x")
 
        # 輸出格式
        ttk.Label(settings_frame, text="輸出格式:").grid(row=0, column=0, sticky="w")
        format_combo = ttk.Combobox(settings_frame, textvariable=self.output_format, 
                                  values=SUPPORTED_FORMATS, state="readonly")
        format_combo.grid(row=0, column=1, padx=5, sticky="ew")
 
        # 輸出路徑
        ttk.Label(settings_frame, text="輸出路徑:").grid(row=1, column=0, sticky="w")
        ttk.Entry(settings_frame, textvariable=self.output_path).grid(row=1, column=1, columnspan=2, padx=5, sticky="ew")
        ttk.Button(settings_frame, text="瀏覽...", command=self.select_output_path).grid(row=1, column=3, padx=5)
 
        # 質(zhì)量設(shè)置
        ttk.Label(settings_frame, text="圖片質(zhì)量 (1-100):").grid(row=2, column=0, sticky="w")
        ttk.Scale(settings_frame, from_=1, to=100, variable=self.quality, 
                 command=lambda v: self.quality.set(int(float(v)))).grid(row=2, column=1, padx=5, sticky="ew")
        ttk.Label(settings_frame, textvariable=self.quality).grid(row=2, column=2, padx=5)
 
        # 尺寸調(diào)整
        resize_frame = ttk.Frame(settings_frame)
        resize_frame.grid(row=3, column=0, columnspan=3, sticky="ew", pady=5)
        ttk.Checkbutton(resize_frame, text="調(diào)整尺寸", variable=self.resize_enabled).pack(side="left", padx=5)
        ttk.Entry(resize_frame, textvariable=self.new_width, width=5).pack(side="left", padx=5)
        ttk.Label(resize_frame, text="x").pack(side="left")
        ttk.Entry(resize_frame, textvariable=self.new_height, width=5).pack(side="left", padx=5)
        ttk.Checkbutton(resize_frame, text="保持比例", variable=self.keep_ratio).pack(side="left", padx=5)
 
        # 其他選項
        ttk.Checkbutton(settings_frame, text="保留元數(shù)據(jù)", variable=self.keep_metadata).grid(row=4, column=0, columnspan=3, sticky="w")
 
        # 操作按鈕
        button_frame = ttk.Frame(self.root)
        button_frame.pack(padx=10, pady=10, fill="x")
 
        ttk.Button(button_frame, text="開始轉(zhuǎn)換", command=self.start_conversion).pack(side="right", padx=5)
        ttk.Button(button_frame, text="退出", command=self.root.destroy).pack(side="right", padx=5)
 
    def add_files(self):
        files = filedialog.askopenfilenames(
            filetypes=[("圖片文件", "*.jpg *.jpeg *.png *.bmp *.gif *.tiff *.webp")]
        )
        if files:
            self.selected_files.extend(files)
            self.update_file_list()
 
    def add_folder(self):
        folder = filedialog.askdirectory()
        if folder:
            for ext in ("*.jpg", "*.jpeg", "*.png", "*.bmp", "*.gif", "*.tiff", "*.webp"):
                self.selected_files.extend(Path(folder).glob(ext))
            self.update_file_list()
 
    def clear_files(self):
        self.selected_files = []
        self.file_list.delete(0, tk.END)
 
    def update_file_list(self):
        self.file_list.delete(0, tk.END)
        for f in self.selected_files:
            self.file_list.insert(tk.END, str(f))
 
    def select_output_path(self):
        path = filedialog.askdirectory()
        if path:
            self.output_path.set(path)
 
    def start_conversion(self):
        if not self.selected_files:
            messagebox.showwarning("警告", "請先選擇要轉(zhuǎn)換的圖片文件!")
            return
    
        output_path = self.output_path.get() or os.path.dirname(self.selected_files[0])
        output_format = self.output_format.get()
    
        # 創(chuàng)建輸出目錄(如果不存在)
        os.makedirs(output_path, exist_ok=True)
    
        for file_path in self.selected_files:
            try:
                img = Image.open(file_path)
                
                # 處理尺寸調(diào)整
                if self.resize_enabled.get():
                    width = self.new_width.get()
                    height = self.new_height.get()
                    original_width, original_height = img.size
 
                    if self.keep_ratio.get():
                        if width > 0 and height == 0:
                            ratio = width / original_width
                            height = int(original_height * ratio)
                        elif height > 0 and width == 0:
                            ratio = height / original_height
                            width = int(original_width * ratio)
                        else:
                            ratio = min(width/original_width, height/original_height)
                            width = int(original_width * ratio)
                            height = int(original_height * ratio)
                    
                    if width > 0 and height > 0:
                        img = img.resize((width, height), Image.Resampling.LANCZOS)
    
                # 構(gòu)建輸出路徑
                filename = os.path.splitext(os.path.basename(file_path))[0]
                output_file = os.path.join(output_path, f"{filename}.{output_format.lower()}")
    
                # 檢查文件是否存在并提示
                if os.path.exists(output_file):
                    response = messagebox.askyesnocancel(
                        "文件已存在",
                        f"目標文件已存在:\n{output_file}\n\n"
                        "請選擇操作:\n"
                        "? 是:覆蓋當(dāng)前文件\n"
                        "? 否:跳過當(dāng)前文件\n"
                        "? 取消:中止全部轉(zhuǎn)換",
                        parent=self.root
                    )
                    
                    if response is None:  # 取消
                        return
                    if not response:      # 跳過
                        continue
    
                # 構(gòu)建輸出路徑
                filename = os.path.splitext(os.path.basename(file_path))[0]
                output_file = os.path.join(output_path, f"{filename}.{output_format.lower()}")
 
                # 保存參數(shù)
                save_args = {'format': output_format}
                if output_format == 'JPEG':
                    save_args['quality'] = self.quality.get()
                    save_args['optimize'] = True
                elif output_format == 'PNG':
                    save_args['compress_level'] = 9 - int(self.quality.get() / 11.1)
 
                # 保存圖片
                if not self.keep_metadata.get():
                    img.info.pop('exif', None)
                img.save(output_file, **save_args)
    
            except Exception as e:
                messagebox.showerror("錯誤", f"處理文件 {file_path} 時出錯:\n{str(e)}")
                continue
    
        messagebox.showinfo("完成", "圖片轉(zhuǎn)換完成!")
 
if __name__ == "__main__":
    root = tk.Tk()
    app = ImageConverterApp(root)
    root.mainloop()

到此這篇關(guān)于Python調(diào)用PIL庫實現(xiàn)圖片格式轉(zhuǎn)換工具的文章就介紹到這了,更多相關(guān)Python PIL圖片格式轉(zhuǎn)換內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • ML神器:sklearn的快速使用及入門

    ML神器:sklearn的快速使用及入門

    這篇文章主要介紹了ML神器:sklearn的快速使用及入門,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-07-07
  • Python實現(xiàn)將多個文件的名稱或后綴名由大寫改為小寫

    Python實現(xiàn)將多個文件的名稱或后綴名由大寫改為小寫

    這篇文章主要介紹了如何基于Python語言實現(xiàn)將多個文件的名稱或后綴名由大寫字母修改為小寫,文中的示例代碼講解詳細,感興趣的可以了解下
    2023-09-09
  • Python實現(xiàn)彈球小游戲的示例代碼

    Python實現(xiàn)彈球小游戲的示例代碼

    這篇文章主要為大家詳細介紹了Python如何實現(xiàn)簡單的彈球小游戲,文中講解非常細致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2022-11-11
  • Python日志模塊logging簡介

    Python日志模塊logging簡介

    這篇文章主要介紹了Python日志模塊logging簡介,本文講解了Logger、Handler、Formatter、日志配置管理、通過文件配置管理日志等內(nèi)容,需要的朋友可以參考下
    2015-04-04
  • python腳本調(diào)用iftop 統(tǒng)計業(yè)務(wù)應(yīng)用流量的思路詳解

    python腳本調(diào)用iftop 統(tǒng)計業(yè)務(wù)應(yīng)用流量的思路詳解

    這篇文章主要介紹了python腳本調(diào)用iftop 統(tǒng)計業(yè)務(wù)應(yīng)用流量的思路詳解,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-10-10
  • Python中如何使用sqlite3操作SQLite數(shù)據(jù)庫詳解

    Python中如何使用sqlite3操作SQLite數(shù)據(jù)庫詳解

    這篇文章主要介紹了Python中SQLite數(shù)據(jù)庫的使用,包括連接數(shù)據(jù)庫、創(chuàng)建表、數(shù)據(jù)增刪改查、事務(wù)管理和參數(shù)化查詢等,并提供了操作示例,需要的朋友可以參考下
    2025-03-03
  • python 利用已有Ner模型進行數(shù)據(jù)清洗合并代碼

    python 利用已有Ner模型進行數(shù)據(jù)清洗合并代碼

    今天小編就為大家分享一篇python 利用已有Ner模型進行數(shù)據(jù)清洗合并代碼,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • python numpy 顯示圖像陣列的實例

    python numpy 顯示圖像陣列的實例

    今天小編就為大家分享一篇python numpy 顯示圖像陣列的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-07-07
  • python 發(fā)送qq郵件的示例

    python 發(fā)送qq郵件的示例

    這篇文章主要介紹了python 發(fā)送qq郵件的示例,幫助大家更好的理解和學(xué)習(xí)使用python,感興趣的朋友可以了解下
    2021-03-03
  • Python格式化輸出%s和%d

    Python格式化輸出%s和%d

    本篇文章主要介紹了Python格式化輸出%s和%d的實例案例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-05-05

最新評論