python 使用tkinter,pydub,AudioSegment把FLAC,ogg, wav等格式音频转换 ...

打印 上一主题 下一主题

主题 850|帖子 850|积分 2550

直接上效果图


可以使用python的pysimplegui-exemaker打包一个exe步伐
把FLAC,WAV,OGG等音频转MP3音频

   在音频处置惩罚领域,我们经常需要对音频文件进行各种操纵,例如剪辑、合并、转码等。而 Python 中的 AudioSegment 是一个强大的库,可以帮助我们处置惩罚各种音频格式的文件。
1.支持的格式
  

  • AudioSegment 是 pydub 库的一部分,通过使用 ffmpeg 库来实现对音频文件格式的支持。因此,AudioSegment 实际上支持所有 ffmpeg 支持的音频格式。
   2.下面是一些常见的音频格式:
  

  • WAV (Waveform Audio File Format,无损音频文件格式)
  • MP3 (MPEG Audio Layer 3,有损音频文件格式)
  • FLAC (Free Lossless Audio Codec,无损音频文件格式)
  • OGG (Ogg Vorbis,开放音频文件格式)
  • AAC (Advanced Audio Coding,高级音频编码格式)
   当然,除了上述格式外,AudioSegment 还支持其他许多音频格式,例如 AMR、WMA、AIFF 等。具体介绍及使用百度自了解
  上代码:python文件为audioConvert.py

  1. """
  2.     把FLAC,WAV,OGG等音频转MP3音频
  3.     AudioSegment 支持哪个格式
  4.     在音频处理领域,我们经常需要对音频文件进行各种操作,例如剪辑、合并、转码等。而 Python 中的 AudioSegment 是一个强大的库,可以帮助我们处理各种音频格式的文件。
  5.     支持的格式
  6.     AudioSegment 是 pydub 库的一部分,通过使用 ffmpeg 库来实现对音频文件格式的支持。因此,AudioSegment 实际上支持所有 ffmpeg 支持的音频格式。下面是一些常见的音频格式:
  7.     WAV (Waveform Audio File Format,无损音频文件格式)
  8.     MP3 (MPEG Audio Layer 3,有损音频文件格式)
  9.     FLAC (Free Lossless Audio Codec,无损音频文件格式)
  10.     OGG (Ogg Vorbis,开放音频文件格式)
  11.     AAC (Advanced Audio Coding,高级音频编码格式)
  12.     当然,除了上述格式外,AudioSegment 还支持其他许多音频格式,例如 AMR、WMA、AIFF 等。
  13. """
  14. import os
  15. import re
  16. from pydub import AudioSegment
  17. from pydub.utils import mediainfo
  18. from queue import Queue
  19. import threading
  20. class AudioConverterMp3:
  21.     """
  22.     :param source: 源文件夹
  23.     :param target: 输出文件夹
  24.     :param source_type: 源格式
  25.     :param target_type: 输出格式
  26.     :param printlog: 输出日志对象方法 没有传值用默认print否则printlog(message)
  27.     """
  28.     def __init__(self, source='F:\夸克网盘\Flac音乐+歌词', target='F:\夸克网盘\MP3音乐', source_type='flac', target_type='mp3', printlog=print):
  29.         self.source = source  # 源文件夹
  30.         self.target = target  # 输出文件夹
  31.         self.source_type = source_type  # 源格式
  32.         self.target_type = target_type  # 输出格式
  33.         self.printlog = printlog  # 输出日志对象方法 没有传值用默认print否则printlog(message)
  34.         self.count = 0  # 待转化音频文件总数
  35.         self.queue = Queue()  # 队列,为什么要用队列?因为如果不用队列,多线程处理音频文件时,会出现文件读写冲突,导致程序崩溃
  36.         self.get_audio_files()  # 获取源文件夹下所有flac音频文件
  37.     def get_audio_files(self):
  38.         """
  39.         获取源文件夹下所有音频文件,加入队列
  40.         :return: 文件列表
  41.         """
  42.         for index, file in enumerate([song for song in os.listdir(self.source) if song.endswith('.' + self.source_type)]):
  43.             self.queue.put([index, file])
  44.             self.count += 1
  45.         self.printlog(f'本次待转化音频文件总数:{self.count}个,开始处理...')
  46.     def audio_to_mp3(self, index, source_file):
  47.         """
  48.         把音频转MP3音频
  49.         :param index: 序号
  50.         :param source_file: 源文件
  51.         :return: None
  52.         """
  53.         try:
  54.             song = AudioSegment.from_file(os.path.join(self.source, source_file), self.source_type)  # 读取音频文件
  55.             target_file = source_file[:-len(self.source_type)] + self.target_type  # 输出文件
  56.             tags = mediainfo(os.path.join(self.source, source_file)).get('TAG')  # 输出音频信息(元数据)
  57.             song.export(os.path.join(self.target, target_file), format=self.target_type, bitrate='192k', tags=tags)  # 保存音频为mp3格式
  58.             self.printlog(f"{threading.current_thread().name} 提供 {source_file} 转换:{self.source_type} -> {self.target_type} 完成")  # 输出结果
  59.         except Exception as e:
  60.             error_dir = os.path.join(os.path.dirname(self.source), "error-transfer-song")
  61.             if not os.path.exists(error_dir):
  62.                 os.makedirs(error_dir)
  63.             source_file = re.findall(r"from '(.+?)':", str(e))[0]
  64.             os.rename(os.path.join(self.source, source_file), os.path.join(error_dir, source_file))  # 发生错误时,将源文件移动到error-transfer-song文件夹下
  65.             self.printlog(f'\n\n{source_file} 转码失败,已移动到 {error_dir} 文件夹下\n\n发生错误代码:{e}\n\n')
  66.     def multiple_thread(self):
  67.         """
  68.         多线程处理音频文件
  69.         多线程处理音频文件,使用队列,队列中存放音频文件路径,每个线程从队列中取出一个音频文件进行处理。
  70.         队列的使用,可以保证多个线程同时处理音频文件,避免文件读写冲突。
  71.         :return: None
  72.         """
  73.         while not self.queue.empty():
  74.             index, file = self.queue.get()
  75.             if os.path.exists(os.path.join(self.target, file[:-len(self.source_type)] + self.target_type)):
  76.                 self.printlog(f'{index + 1}/{self.count}: {file} 已转换,跳过...')
  77.                 continue
  78.             self.audio_to_mp3(index, file)
  79.     # 运行程序
  80.     def run(self):
  81.         """
  82.         运行程序
  83.         多线程处理,根据CPU核心数启动线程数,可以充分利用CPU资源,提高处理速度。
  84.         :return: None
  85.         """
  86.         tasks = []
  87.         for i in range(os.cpu_count()):
  88.             t = threading.Thread(target=self.multiple_thread, name=f'线程-{i}')
  89.             t.start()
  90.             tasks.append(t)
  91.         for t in tasks:
  92.             t.join()
  93.         self.printlog('\n全部文件处理完成!\n')
  94. # 程序测试用入口,,当文件直接运行时,即非模块导入调用,执行以下代码
  95. if __name__ == '__main__':
  96.     audio_to_mp3 = AudioConverterMp3()
  97.     audio_to_mp3.run()
复制代码
python图形界面教程(tkinter)

   主要用到 tkinderLabel标签显示文体,Radiobuttio 单选按钮选择源文件格式,tkinter.filedialog.askdirectory 选择文件目次,滚动文本ScrolledText等具体介绍及使用百度自了解。
    在Python的Tkinter GUI中动态输出日志时,如果碰到卡顿题目,通常是因为主事件循环被过度占用,导致界面无法实时更新。为了办理这个题目,可以使用多线程或者异步编程的方式,让日志输出在另一个线程中执行,从而不影响主事件循环的运行。
  上代码:python文件为main.py(主文件)

  1. # python图形界面:   
  2. from threading import Thread
  3. from tkinter import *
  4. from tkinter import messagebox
  5. from tkinter.filedialog import askdirectory
  6. from tkinter.scrolledtext import ScrolledText
  7. from audioConvert import AudioConverterMp3
  8. class AudioToMP3UI(object):
  9.     """
  10.     把音频转MP3格式音频
  11.     1. 选择源音频文件路径和目标音频文件路径
  12.     2. 点击“开 始 转 码”按钮,开始转码
  13.     3. 转码完成后,会弹出提示框,提示转码完成
  14.     4. 转码完成后,可以到目标音频文件路径下查看转码后的音频文件
  15.     5. 日志信息会实时显示在滚动文本框中
  16.     """
  17.     def __init__(self):
  18.         self.et_source = None  # 源文件路径输入框
  19.         self.et_target = None  # 目标文件路径输入框
  20.         self.scrolltext = None  # 滚动文本框
  21.         self.thread = None  # 线程对象
  22.         self.root = Tk()  # 创建GUI窗口
  23.         self.root.title("把音频转MP3格式音频")  # 设置窗口标题
  24.         width = 600
  25.         height = 500
  26.         x = self.root.winfo_screenwidth() // 2 - width // 2  # 计算窗口宽度居中位置
  27.         y = self.root.winfo_screenheight() // 2 - height // 2  # 计算窗口高度居中位置
  28.         self.root.geometry(f"{width}x{height}+{x}+{y}")  # 设置窗口大小和位置
  29.         self.source_type = StringVar()  # 源文件格式选择变量
  30.         self.frame_type = Frame(self.root)  # 创建Frame容器
  31.         self.frame_type.pack()  # 放置Frame容器
  32.         self.frame_path = Frame(self.root)  # 创建Frame容器
  33.         self.frame_path.pack()  # 放置Frame容器
  34.         self.frame_info = Frame(self.root)  # 创建Frame容器
  35.         self.frame_info.pack()  # 放置Frame容器 AMR、WMA、AIFF
  36.     # 选择源文件格式
  37.     def create_source_type_ui(self):
  38.         lb_type = Label(self.frame_type, text="源文件格式:")  # 创建标签
  39.         lb_type.grid(column=0, row=0, padx=8, pady=4)  # 放置标签
  40.         rb_type1 = Radiobutton(self.frame_type, text="wav", variable=self.source_type, value="wav")  # 创建单选按钮
  41.         rb_type1.grid(column=1, row=0, padx=3)  # 放置标签
  42.         rb_type2 = Radiobutton(self.frame_type, text="mp3", variable=self.source_type, value="mp3")  # 创建单选按钮
  43.         rb_type2.grid(column=2, row=0, padx=3)  # 放置标签
  44.         rb_type3 = Radiobutton(self.frame_type, text="flac", variable=self.source_type, value="flac")  # 创建单选按钮
  45.         rb_type3.grid(column=3, row=0, padx=3)  # 放置标签
  46.         rb_type4 = Radiobutton(self.frame_type, text="ogg", variable=self.source_type, value="ogg")  # 创建单选按钮
  47.         rb_type4.grid(column=4, row=0, padx=3)  # 放置标签
  48.         rb_type5 = Radiobutton(self.frame_type, text="aac", variable=self.source_type, value="aac")  # 创建单选按钮
  49.         rb_type5.grid(column=5, row=0, padx=3)  # 放置标签
  50.         rb_type6 = Radiobutton(self.frame_type, text="amr", variable=self.source_type, value="amr")  # 创建单选按钮
  51.         rb_type6.grid(column=6, row=0, padx=3)  # 放置标签
  52.         rb_type7 = Radiobutton(self.frame_type, text="wma", variable=self.source_type, value="wma")  # 创建单选按钮
  53.         rb_type7.grid(column=7, row=0, padx=3)  # 放置标签
  54.         rb_type8 = Radiobutton(self.frame_type, text="aiff", variable=self.source_type, value="aiff")  # 创建单选按钮
  55.         rb_type8.grid(column=8, row=0, padx=3)  # 放置标签
  56.         self.source_type.set("flac")  # 设置默认选择flac格式
  57.     def select_type_command(self):
  58.         print("选择源文件格式:" + self.source_type.get())  # 打印选择的源文件格式
  59.         return self.source_type.get()
  60.     # 设置源文件路径和目标文件路径输入框和打开文件目录按钮
  61.     def create_source_target_dir_ui(self):
  62.         # 设置源文件路径输入框和打开文件目录按钮
  63.         lb_source = Label(self.frame_path, text="源音频文件路径:")  # 创建标签
  64.         lb_source.grid(column=0, row=0, pady=20)  # 放置标签
  65.         self.et_source = Entry(self.frame_path, width=60, bd=5)  # 创建源目录输入框
  66.         self.et_source.grid(column=1, row=0, padx=5)  # 放置输入框
  67.         btn_source = Button(self.frame_path, text="点选", command=self.select_source_dir_command)  # 创建选择源目录按钮
  68.         btn_source.grid(column=2, row=0)  # 放置按钮
  69.         # 设置目标文件路径输入框和打开文件目录按钮
  70.         lb_target = Label(self.frame_path, text="导出音频文件路径:")  # 创建标签
  71.         lb_target.grid(column=0, row=1, pady=0)  # 放置标签
  72.         self.et_target = Entry(self.frame_path, width=60, bd=5)  # 创建目标目录输入框
  73.         self.et_target.grid(column=1, row=1, padx=5)  # 放置输入框
  74.         btn_target = Button(self.frame_path, text="点选", command=self.select_target_dir_command)  # 创建选择目标目录按钮
  75.         btn_target.grid(column=2, row=1)  # 放置按钮
  76.         btn_start = Button(self.frame_path, text="开 始 转 码", command=self.convert_format)  # 创建开始转码按钮
  77.         btn_start.grid(column=0, row=2, ipadx=80, padx=45, pady=15, columnspan=3)
  78.     # 设置选择源文件目录按钮事件
  79.     def select_source_dir_command(self):
  80.         dirname = askdirectory(title='请选择->源音频文件路径')  # 弹出选择源目录对话框
  81.         if dirname != '':
  82.             self.et_source.delete(0, END)  # 清空输入框内容
  83.             self.et_source.insert(0, dirname)  # 显示选择的目录
  84.         else:
  85.             messagebox.showinfo("提醒", "您没有选择任何目录!")  # 提示用户没有选择目录
  86.     # 设置选择目标文件目录按钮事件
  87.     def select_target_dir_command(self):
  88.         dirname = askdirectory(title='请选择->导出音频文件路径')  # 弹出选择目标目录对话框
  89.         if dirname != '':
  90.             self.et_target.delete(0, END)  # 清空输入框内容
  91.             self.et_target.insert(0, dirname)  # 显示选择的目录
  92.         else:
  93.             messagebox.showinfo("提醒", "您没有选择任何目录!")  # 提示用户没有选择目录
  94.     # 设置滚动文本框
  95.     def create_scrolltext_ui(self):
  96.         self.scrolltext = ScrolledText(self.frame_info, width=78, height=21, padx=10, pady=8)  # 创建滚动文本框
  97.         self.scrolltext.pack()  # 放置滚动文本框
  98.         self.scrolltext.insert(END, "1. 选择源音频文件路径和目标音频文件路径\n"
  99.                                     "2. 点击“开 始 转 码”按钮,开始转码\n"
  100.                                     "3. 转码完成后,会弹出提示框,提示转码完成\n"
  101.                                     "4. 转码完成后,可以到目标音频文件路径下查看转码后的音频文件\n")    # 设置提示信息
  102.     # 设置开始转码按钮事件
  103.     def convert_format(self):
  104.         source_path = self.et_source.get()  # 获取源文件路径
  105.         target_path = self.et_target.get()  # 获取目标文件路径
  106.         if source_path == '' or target_path == '':
  107.             messagebox.showinfo("提醒", "请选择源文件路径和目标文件路径!")    # 提示用户没有选择目录
  108.         else:
  109.             self.scrolltext.insert(END, f'源文件路径:《{source_path}》 \n')  # 显示源文件路径
  110.             self.scrolltext.insert(END, f'目标文件路径:《{target_path}》 \n')  # 显示目标文件路径
  111.             self.scrolltext.insert(END, "开始转码!\n")  # 显示开始转码信息
  112.             self.root.update()  # 更新UI显示
  113.             # 转码代码
  114.             audio_to_mp3 = AudioConverterMp3(source_path, target_path, self.source_type.get(), printlog=self.printlog)  # 创建AudioConverterMp3对象
  115.             # 在Python的Tkinter GUI中动态输出日志时,如果遇到卡顿问题,通常是因为主事件循环被过度占用,导致界面无法及时更新。
  116.             # 为了解决这个问题,可以使用多线程或者异步编程的方式,让日志输出在另一个线程中执行,从而不影响主事件循环的运行。
  117.             self.thread = Thread(target=audio_to_mp3.run)    # 创建线程对象
  118.             self.thread.start()  # 启动线程
  119.     # 输出日志信息,用于显示执行过程打印的日志信息
  120.     def printlog(self, info):
  121.         self.scrolltext.insert(END, info + '\n')  # 显示日志信息
  122.         self.scrolltext.see(END)  # 滚动到最后一行
  123.         self.root.update()  # 更新UI显示
  124.     # 运行GUI
  125.     def run(self):
  126.         self.create_source_type_ui()  # 创建源文件格式选择按钮
  127.         self.create_source_target_dir_ui()  # 创建源文件路径输入框和打开文件目录按钮
  128.         self.create_scrolltext_ui()  # 创建滚动文本框
  129.         self.root.mainloop()    # 运行GUI
  130. if __name__ == '__main__':
  131.     trans_mp3 = AudioToMP3UI()
  132.     trans_mp3.run()
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

勿忘初心做自己

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表