【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_())
相关推荐
minji...25 分钟前
Linux 线程同步与互斥(三) 生产者消费者模型,基于阻塞队列的生产者消费者模型的代码实现
linux·运维·服务器·开发语言·网络·c++·算法
Dxy123931021632 分钟前
Python基于BERT的上下文纠错详解
开发语言·python·bert
wjs20242 小时前
JavaScript 语句
开发语言
cmpxr_3 小时前
【C】局部变量和全局变量及同名情况
c语言·开发语言
小碗羊肉4 小时前
【从零开始学Java | 第三十一篇下】Stream流
java·开发语言
aq55356005 小时前
Laravel10.x重磅升级,新特性一览
android·java·开发语言
报错小能手5 小时前
ios开发方向——swift错误处理:do/try/catch、Result、throws
开发语言·学习·ios·swift
老歌老听老掉牙5 小时前
PyQt5+Qt Designer实战:可视化设计智能参数配置界面,告别手动布局时代!
python·qt
网域小星球5 小时前
C 语言从 0 入门(十七)|结构体指针 + 动态内存 + 文件综合实战
c语言·开发语言·文件操作·结构体指针·动态内存·综合项目