【PyQt5】嵌套多线程数据交互实现

使用场景:

PyQt5 +Python开发的GUI小工具,使用QThread实现前后端数据交互,同时,后端使用2个多线程进行逻辑编写,在线程1里面启动线程2,且线程2需要线程1给传递数据。

python 复制代码
import sys
import time
import threading
import pyaudio
import wave
from PyQt5.QtCore import QThread, pyqtSignal, QMutex, QTimer  
from PyQt5.QtWidgets import (QApplication, QMainWindow, QPushButton, 
                             QTextBrowser, QVBoxLayout, QWidget, QLabel, 
                             QSpinBox, QHBoxLayout)

class AudioRecorderThread(QThread):
    """
    线程2:音频录制线程
    """
    update_signal = pyqtSignal(str)  # 用于发送更新信息的信号
    recording_finished = pyqtSignal(str)  # 录制完成信号
    audio_wav_name=pyqtSignal(str)  # 传递录制的音频文件名
    
    def __init__(self, filename="output.wav", parent=None):
        super().__init__(parent)
        self.filename = filename
        self._is_recording = False
        self._mutex = QMutex()
        self.frames = []
        self.audio = pyaudio.PyAudio()
        
        # 音频参数
        self.format = pyaudio.paInt16
        self.channels = 1
        self.rate = 44100
        self.chunk = 1024

    def run(self):
        """开始录制音频"""
        self._is_recording = True
        self.frames = []
        
        try:
            # 打开音频流
            stream = self.audio.open(
                format=self.format,
                channels=self.channels,
                rate=self.rate,
                input=True,
                frames_per_buffer=self.chunk
            )
            
            self.update_signal.emit("音频录制已开始...")
            
            # 录制循环
            while self.is_recording():
                data = stream.read(self.chunk)
                self.frames.append(data)
                
            # 停止录制
            stream.stop_stream()
            stream.close()
            
            # 保存录音文件
            self.save_audio()
            self.update_signal.emit("音频录制已停止")
            self.recording_finished.emit(self.filename)
            
        except Exception as e:
            self.update_signal.emit(f"录制错误: {str(e)}")

    def is_recording(self):
        """检查是否正在录制(线程安全)"""
        with QMutexLocker(self._mutex):
            return self._is_recording

    def stop_recording(self):
        """停止录制(线程安全)"""
        with QMutexLocker(self._mutex):
            self._is_recording = False

    def save_audio(self):
        """保存录制的音频到文件"""
        try:
            wf = wave.open(self.filename, 'wb')
            wf.setnchannels(self.channels)
            wf.setsampwidth(self.audio.get_sample_size(self.format))
            wf.setframerate(self.rate)
            wf.writeframes(b''.join(self.frames))
            wf.close()
            self.audio_wav_name.emit(self.filename)
            self.update_signal.emit(f"音频已保存到: {self.filename}")
        except Exception as e:
            self.update_signal.emit(f"保存音频错误: {str(e)}")

    def __del__(self):
        """清理资源"""
        self.audio.terminate()


class ControlThread(QThread):
    """
    线程1:控制线程,管理音频录制线程
    """
    update_signal = pyqtSignal(str)  # 用于发送更新信息的信号
    
    def __init__(self, max_duration=10, parent=None):
        super().__init__(parent)
        self._isPause=False #是否暂停
        self.is_runnnin=True #标志位表示线程是否需要退出
        self.finished=False
        self.cond=QWaitCondition()
        self.mutex=QMutex()
        self.max_duration = max_duration  # 最大录制时长(秒)
        self.record_wav_file=None #音频文件的绝对路径
        self.recorder = AudioRecorderThread()
        # 将录制线程的信号连接到控制线程的信号
        self.recorder.update_signal.connect(self.update_signal.emit)
        self.recorder.recording_finished.connect(self.on_recording_finished)
        self.recorder.audio_wav_name.connect(self.get_audio_wav_name)
        self.recording_start_time = 0
    def check_mutex(self):
        self.mutex.lock()
        while self._isPuse:
            self.cond.wait(self.mutex)
        self.mutex.unlock()
    def stop_recording(self):
        """停止录制线程"""
        if self.recorder.is_recording():
            self.recorder.stop_recording()
            # 等待录制线程结束
            self.recorder.wait()

    def on_recording_finished(self, filename):
        """录制完成时的处理"""
        self.update_signal.emit(f"录制完成,文件已保存: {filename}")
        
	def get_audio_wav_name(self,wav_name):
		self.record_wav_file=wav_name
		
    def run(self):
        """启动控制逻辑"""
        self.update_signal.emit("控制线程启动")
        while not self.finished:
            # 如果是暂停状态,则阻塞等待
        	self.check_mutex()
        	#todo: 可填写其他操作步骤
	        # 启动音频录制线程
	        self.recorder.start()
	        self.recording_start_time = time.time()
	        
	        # 检查是否达到停止条件
	        while self.recorder.isRunning():
	            elapsed = time.time() - self.recording_start_time
	            self.update_signal.emit(f"已录制: {elapsed:.1f}秒")
	            
	            # 检查是否达到最大时长条件
	            if elapsed >= self.max_duration:
	                self.update_signal.emit(f"已达到最大录制时长({self.max_duration}秒),停止录制...")
	                self.stop_recording()
	                break
	                
	            # 休眠一段时间再检查
	            self.msleep(500)  # 每500毫秒检查一次
	        #todo: 可填写其他操作步骤    
	        self.update_signal.emit("控制线程结束")

class MainWindow(QMainWindow):
    def __init__(self):
        super().__init__()
        self.control_thread = None
        self.init_ui()

    def init_ui(self):
        self.setWindowTitle('PyQt5 音频录制工具')
        self.setGeometry(100, 100, 600, 400)

        layout = QVBoxLayout()

        # 状态显示
        self.text_browser = QTextBrowser(self)
        layout.addWidget(self.text_browser)

        # 录制时长设置
        duration_layout = QHBoxLayout()
        duration_layout.addWidget(QLabel("录制时长(秒):"))
        self.duration_spinbox = QSpinBox()
        self.duration_spinbox.setRange(1, 3600)  # 1秒到1小时
        self.duration_spinbox.setValue(10)  # 默认10秒
        duration_layout.addWidget(self.duration_spinbox)
        duration_layout.addStretch()
        layout.addLayout(duration_layout)

        # 按钮布局
        button_layout = QHBoxLayout()
        
        self.start_button = QPushButton('开始录制', self)
        self.start_button.clicked.connect(self.start_recording)
        button_layout.addWidget(self.start_button)

        self.stop_button = QPushButton('停止录制', self)
        self.stop_button.clicked.connect(self.stop_recording)
        self.stop_button.setEnabled(False)
        button_layout.addWidget(self.stop_button)

        layout.addLayout(button_layout)

        central_widget = QWidget()
        central_widget.setLayout(layout)
        self.setCentralWidget(central_widget)

    def start_recording(self):
        """启动录制"""
        if self.control_thread is None or not self.control_thread.isRunning():
            max_duration = self.duration_spinbox.value()
            self.control_thread = ControlThread(max_duration)
            self.control_thread.update_signal.connect(self.update_text_browser)
            self.control_thread.start()
            
            self.start_button.setEnabled(False)
            self.stop_button.setEnabled(True)
            self.text_browser.append("已启动录制线程...")

    def stop_recording(self):
        """停止录制"""
        if self.control_thread and self.control_thread.isRunning():
            self.text_browser.append("正在停止录制...")
            self.control_thread.stop_recording()
            # 等待控制线程结束
            self.control_thread.wait()
            
            self.start_button.setEnabled(True)
            self.stop_button.setEnabled(False)

    def update_text_browser(self, message):
        """更新状态显示"""
        self.text_browser.append(message)

    def closeEvent(self, event):
        """确保窗口关闭时所有线程已停止"""
        if self.control_thread and self.control_thread.isRunning():
            self.control_thread.stop_recording()
            if not self.control_thread.wait(2000):  # 等待最多2秒
                self.control_thread.terminate()  # 强制终止
                self.control_thread.wait()
        event.accept()


if __name__ == '__main__':
    app = QApplication(sys.argv)
    
    # 检查pyaudio是否可用
    try:
        p = pyaudio.PyAudio()
        p.terminate()
    except Exception as e:
        print(f"PyAudio初始化错误: {e}")
        print("请确保已安装PyAudio和PortAudio")
        sys.exit(1)
    
    window = MainWindow()
    window.show()
    sys.exit(app.exec_())
相关推荐
我滴老baby1 分钟前
工具调用全景解析从Function Calling到MCP协议的完整实践
开发语言·人工智能·python·架构·fastapi
小李子呢02111 分钟前
前端八股JS---Map / Set / WeakMap / WeakSet
开发语言·前端·javascript
feifeigo1233 分钟前
自适应大邻域搜索(ALNS)算法的MATLAB 实现
开发语言·算法·matlab
繁星蓝雨11 分钟前
Qt多界面创建的优化问题(main函数或主界面中创建?)—————附带详细方法
c++·qt·架构·多界面管理
沐知全栈开发13 分钟前
API 类别 - 实用工具
开发语言
Cx330❀19 分钟前
Qt 入门指南:从零搭建开发环境到第一个图形界面程序
xml·大数据·开发语言·网络·c++·人工智能·qt
SilentSamsara20 分钟前
装饰器基础:从闭包到装饰器的自然演变
开发语言·前端·vscode·python·青少年编程·pycharm
今天长肉了吗1 小时前
风控指标平台实战:大数据量下如何设计分批处理
开发语言·数据库·python
ch.ju1 小时前
Java programming(The third edition) Chapter Two——Null return value
java·开发语言
折哥的程序人生 · 物流技术专研1 小时前
第3篇:为何要配置环境变量?
java·开发语言·后端·面试