QThread无法正确使用QTimer

时间:2016-08-24 21:05:18

标签: pyqt pyqt5 qthread qtimer

在我的应用程序中,我有一个自定义QThread负责与后端通信,并使用url和run()方法中的数据调用实用程序函数:

class SomeThread(QtCore.QThread):
  def __init__(self, parent=None...):
    QtCore.QThread.__init__(self, parent)
    ...
  def run(self):
    final_desired_content = some_utility_method(url, data ...)
    # emitting success with final_desired_content

在实用程序方法中,我正在进行http POST,获取响应,解析响应,并最终将所需信息传递给上面的线程变量final_desired_content。在我传回信息之前,我正在解析一些我不想返回的信息,并希望存储在SomeClass单例实例中:

def some_utility_method( ... ):
  ...
  return response_parsing(response)

def response_parsing(response):
  ...
  some_file.SomeClass.instance().setNewData(otherData)
  return mainParsedData

因为可能有几个线程在几秒钟内(特别是在应用程序启动期间)联系BE,我想在通过之前阻止数据写入(可以忽略我们忽略的数据):

class SomeClass(QtCore.QObject):

  _instance = None

  @classmethod
  def instance(klass):
    if not klass._instance:
      klass._instance = SomeClass()
    return klass._instance

  def __init__(self):
    QtCore.QObject.__init__(self)
    self._recentlyUpdatedTimer = QtCore.QTimer()
    self._recentlyUpdatedTimer.setSingleShot(True)
    self._recentlyUpdatedTimer.timeout.connect(self._setOkToUpdateCB)
    self._storedData = None
    self._allowUpdate = True

  def _setOkToUpdateCB(self):
    self._allowUpdate = True

  def setNewData(self, newData):
    if self._allowUpdate:
      print "UPDATING!"
      self._allowUpdate = False
      self._storedData = newData
      self._recentlyUpdatedTimer.start(<some_time>)
    else:
      print "BLOCKED!" # ok to ignore newData

问题是这次成功更新一次,然后在第二次更新成功后,我收到此错误:QObject :: startTimer:无法从另一个线程启动定时器

根据我所知和读的线程,QThread中的run()是另一个线程,可能不知道主线程中发生了什么。 调试时,看起来计时器仍在运行,即使它设置为singleShot。

我将不胜感激任何建议:)

1 个答案:

答案 0 :(得分:0)

我通过不使用计时器解决了这个问题。 我最终做的是:

class SomeClass(QtCore.QObject):
  ...
  DATA_EXPIRE_THRESHOLD = 3
  ...
  def __init__(self):
    ...
    self._counter = 0 # Just for debugging
    self._dataExpireTime = None
    self._data = None

  def setNewData(self, new_data):
    self._counter += 1
    if self._dataExpireTime is None or self._isExpired():
      print "self._isExpired(): [%s], counter: [%s]" % (self._isExpired(), self.counter)
      self._data = new_data
      self._dataExpireTime = time.time() + self.DATA_EXPIRE_THRESHOLD

  def _isExpired(self):
    return time.time() >= self._dataExpireTime 

所以将阈值设置为3秒,输出为:

self._isExpired(): [True], counter: [1] 
self._isExpired(): [True], counter: [2]
self._isExpired(): [True], counter: [6]
self._isExpired(): [True], counter: [15]
self._isExpired(): [True], counter: [17]
self._isExpired(): [True], counter: [18]

我试图使用互斥锁,但它与计时器有类似的问题。 我仍然感谢对这些问题的解释或建议。