【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_())
相关推荐
SHolmes185414 分钟前
给定某日的上班时间段,计算当日的工作时间总时长(Python)
开发语言·前端·python
咖啡の猫22 分钟前
Python字典元素的增、删、改操作
java·开发语言·python
Lucky小小吴30 分钟前
JAVA漫谈反序列化篇——笔记
java·开发语言·笔记
ytttr8731 小时前
基于 C# WinForm 实现的 电影院售票系统
开发语言·c#
Tony Bai1 小时前
Goroutine “气泡”宇宙——Go 并发模型的新维度
开发语言·后端·golang
Larry_Yanan1 小时前
Qt多进程(二)QProcess+stdio
开发语言·qt
码农葫芦侠1 小时前
Qt 跨线程内存管理陷阱:QSharedPointer、deleteLater() 与 QPointer 的致命组合
开发语言·数据库·qt
d111111111d1 小时前
C语言中,malloc和free是什么,在STM32中使用限制是什么,该如何使用?
c语言·开发语言·笔记·stm32·单片机·嵌入式硬件·学习
网安_秋刀鱼1 小时前
【java安全】shiro鉴权绕过
java·开发语言·安全
白昼流星!2 小时前
C++内存四区与new操作符详解
开发语言·c++