阻塞QThread会阻塞GUI

2 投票
1 回答
1433 浏览
提问于 2025-04-17 11:13

我在做一个简单的聊天程序,使用了一个C语言的库,并通过boost::python进行了封装。

界面是用PyQT写的。接收消息是通过一个阻塞的调用来实现的,也就是说在等待消息的时候,程序会停下来不动。为了让界面能够独立刷新,我把通信部分放在了一个QThread里。

虽然我以为界面和通信应该是独立的,但实际上界面反应非常慢,似乎只有在收到消息的时候才会更新。

#!/usr/bin/env python

import sys

from PyQt4.QtCore import *
from PyQt4.QtGui import *

import pynetcom2
import time


class NetCom(QThread):

  def __init__(self):
    QThread.__init__(self)
    self.client = pynetcom2.Client()
    self.client.init('127.0.0.1', 4028)
    self.client.provide('myChat', 1)
    self.client.subscribe('myChat', 100)

  def run(self):
    while (1):
      print "Waiting for message..."
      text = self.client.recvStr('myChat', True)
    return



class Netchat(QMainWindow):

    def __init__(self, argv):

        if (len(argv) != 2):
            print "Usage: %s <nickname>" %(argv[0])
            sys.exit(1)
        self.nickname = argv[1]
        print "Logging in with nickname '%s'" %(self.nickname)

        super(Netchat, self).__init__()
        self.setupUI()

        rect = QApplication.desktop().availableGeometry()
        self.resize(int(rect.width() * 0.3), int(rect.height() * 0.6))
        self.show()

        self.com = NetCom()
        self.com.start()

    def setupUI(self):
        centralWidget = QWidget()
        self.setCentralWidget(centralWidget)

        self.testList = QListWidget()

        mainLayout = QHBoxLayout()
        mainLayout.addWidget(self.testList)
        centralWidget.setLayout(mainLayout)

if __name__ == "__main__":
  app = QApplication(sys.argv)
  netchat = Netchat(sys.argv)
  app.exec_()

1 个回答

6

这可能是因为著名的全局解释器锁(GIL)导致的。Python不允许两个线程同时执行Python代码。在你的C函数中,如果你想让你的图形界面代码并行运行,你必须明确地释放和重新获取这个锁。

这个问题在Python的C API文档中有详细说明:线程状态和全局解释器锁

简单来说,你需要在你的C扩展中使用以下宏:

Py_BEGIN_ALLOW_THREADS

// Your expensive computation goes here.

Py_END_ALLOW_THREADS

撰写回答