Qt——线程类QThread
本文主要介绍Qt中线程类QThread的用法,参考(翻译+修改)了一篇文章:PyQt: Threading Basics Tutorial,虽然使用的是PyQt,但与C++中Qt的用法大同小异,不必太在意语言的差异。
在这篇文章中,我将写一个获取热点新闻的程序(使用新闻网站reddit.com的api),每隔2秒发送一个关键字,从服务器获得与该关键字相关的一条热点新闻。
我们的目标是实现以下几个功能:
- 用户在输入框中输入n个关键字,以英文的逗号,隔开
- 用一个搜索结果列表来呈现所获得的新闻标题
- 使用进度条更新已获得的新闻数目
- 用户随时可以停止获取数据
界面设计如下图:
上面是一个关键字输入框QLineEdit,中间使用QListWidget呈现获得的数据,下面是QProgressBar更新进度,最下面有一个停止按钮和一个开始按钮。
一、代码片段
1.新闻获取部分
我们使用接口https://www.reddit.com/r/keyword.json?limit=1从服务器获取数据。
import json
import time
import requests agent = 'Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.8 Safari/537.36'
headers = {
'User-Agent': agent
} def get_top_post(subreddit):
#从服务器获取数据
url = "https://www.reddit.com/r/{}.json?limit=1".format(subreddit)
try:
restext = requests.get(url, headers=headers)
data = json.loads(restext.text)
top_post = data['data']['children'][0]['data']
except Exception as e:
print(e)
return '错误数据'
return "'{title}' by {author} in {subreddit}".format(**top_post) def get_top_from_subreddits(subreddits):
for subreddit in subreddits:
yield get_top_post(subreddit)
time.sleep(2) if __name__ == '__main__':
for post in get_top_from_subreddits(['python', 'php', 'learnpython']):
print(post)#输出结果
上面是获取并处理新闻数据的程序。需要注意的是其中 time.sleep(2) ,之所以每次发送请求要隔两秒,是因为服务器出于性能考虑,只允许每2秒发送一次请求,否则可能会得到错误的数据。在这里有3个关键字,python、php、learnpython,所以整个过程持续了大约6秒。
不必在意其中实现的细节,因为本文的重点是线程,而不是获取数据。
2.基本界面
我们可以在代码中实现所有控件和布局;也可以用Qt Designer设计好,然后使用命令 pyuic5 -o yourui.py yourui.ui 生成界面代码(具体可以参考该文:点我)。
在这里,我用的是第一个方法:
def initUI(self):
self.setWindowTitle('QThread Study') keywordLbl = QLabel('关键字(以逗号,隔开):')
self.keywordEdit = QLineEdit()
hrLayout = QHBoxLayout()
hrLayout.addWidget(keywordLbl)
hrLayout.addWidget(self.keywordEdit) resultLbl = QLabel('搜索结果:')
self.resultList = QListWidget()
vrLayout = QVBoxLayout()
vrLayout.addWidget(resultLbl)
vrLayout.addWidget(self.resultList) self.searchProgBar = QProgressBar()
self.searchProgBar.setValue(0)
self.stopBtn = QPushButton('停止')
self.stopBtn.setEnabled(False)
self.startBtn = QPushButton('开始')
hrLayout1 = QHBoxLayout()
hrLayout1.addWidget(self.stopBtn)
hrLayout1.addWidget(self.startBtn) vrLayout1 = QVBoxLayout(self)
vrLayout1.addLayout(hrLayout)
vrLayout1.addLayout(vrLayout)
vrLayout1.addWidget(self.searchProgBar)
vrLayout1.addLayout(hrLayout1)
二、未使用多线程
如果没有使用多线程,你可能会这么做:写好新闻获取的代码、写好界面代码,接下来简单地调用函数处理数据。这么做可以,但所有工作都在单独的GUI线程中完成,所以执行函数获取新闻时,你的程序将会被“冻结”住。
就像这样:
- 主线程被锁住
- 直到程序执行结束,搜索结果列表才会更新
- 输入框以及其它界面中的元素都无法使用
- 一旦函数开始执行,就没法停止获取数据
下面是主要代码(点击开始按钮 - 进入槽函数 - 获取新闻数据):
class ThreadTestUI(QWidget):
def __init__(self, parent = None):
super().__init__(parent)
self.initUI()
#建立信号槽连接
self.startBtn.clicked.connect(self.startBtnClicked) def startBtnClicked(self):
subreddit_list = str(self.keywordEdit.text()).split(',')
if subreddit_list == ['']:
print('没有搜索内容')
return
self.resultList.clear()
for post in self.get_top_from_subreddits(subreddit_list):
self.resultList.addItem(post)
三、使用多线程
没有使用多线程将导致程序卡住,体验很差,下面将使用QThread类重写我们的代码。
首先要做的就是写一个线程,这个线程与之前新闻获取部分 get_top_post 和 get_top_from_subreddits 做相同的事,每当获得新数据就立即更新界面,而且允许用户点击“停止”按钮停止获取数据。
1.QThread的基本结构
QThread类很简单,它的整体结构如下:
from PyQt4.QtCore import QThread class YourThreadName(QThread): def __init__(self):
QThread.__init__(self) def __del__(self):
self.wait() def run(self):
# your logic here
你可以通过给构造方法 __init__ 添加参数,将数据传给线程。
在 run 方法中处理你的数据。
注意不能直接调用run方法,而是通过 start 方法间接调用它,否则界面仍有可能被“冻结”住。
接下来是使用上面你定义的线程:
self.myThread = YourThreadName()
self.myThread.start()
如此,在run方法中写的代码得以执行,可以使用像isRunning这样的方法检测线程是否正在运行。
你可能会经常用到这些QThread的方法: quit 、 start 、 terminate 、 isFinished 、 isRunning 。
还有QThread的这些信号: finished 、 started 、 terminated 。
2.我们的程序
介绍完QThread类,下面回到我们的新闻获取程序。
我们可以很容易地将获取新闻的代码移到QThread类,除了修改run方法,其它地方基本保持原样。
另一个小的变化是,需要将新闻关键字的列表传到线程类中,从而在run方法中使用这些关键字。
def setSubReddit(self, subReddit):
self.subreddits = subReddit def run(self):
for subreddit in self.subreddits:
top_post = self._get_top_post(subreddit)
self.sleep(2)
_get_top_post 方法是从之前的新闻获取代码直接复制过来的,在run方法中遍历之前设置的关键字subreddits。
主界面类:
self.testThread.setSubReddit(subreddit_list)
self.testThread.start()
OK,程序将在单独的线程中运行,然后根据关键字获取所有热点新闻。
但是,界面中的元素还没有得到更新,没有反馈给用户,所以我们还需做些什么。
当然,不能简单地在线程类中这么写: self.searchProgBar.setValue(int) ,因为它指向QThread对象,而不是UI对象。
在数据处理线程和UI线程之间沟通的正确方法是使用“信号”。
四、信号
数据获取线程在背后运行,主界面线程需要获得数据(比如新闻标题),从而更新界面元素(比如进度条和新闻列表)
下面先讲一下Pyqt的信号,它与C++中信号槽连接有所不同。
1.内建信号
获取数据结束之后需要通知用户,我们将使用一个所有QThread实例都有的信号。
首先写一个线程结束后我们想要执行的代码,比如打印一条信息,我们在主界面类中这么写:
def threadFinished(self):
print('获取结束')
接下来是信号的连接,将QThread实例发出的信号与我们线程结束后打印信息的函数连接起来:
self.testThread = GetPostThread()
self.testThread.finished.connect(self.threadFinished)
内建信号与槽函数的连接很直接,自定义信号与之唯一的不同就是,我们首先需要在QThread类中定义一个信号,在主线程中的写法是一样的。
所以接下来——
2.自定义信号
想要像内建信号一样使用自定义信号,首先需要定义它们,在QThread类中定义信号:
postSignal = pyqtSignal(str)
注意:定义的信号有一个参数,类型是字符串str。
run方法中处理并获得数据,然后通过信号将其发出:
def run(self):
for subreddit in self.subreddits:
top_post = self._get_top_post(subreddit)
self.postSignal.emit(top_post)
self.sleep(2)
主线程获得信号,并将它与信号处理函数(槽函数)相连接:
self.testThread.postSignal.connect(self.getPostSlot)
信号发出时带有一个字符串参数(在这里是新闻的标题),定义信号处理函数时也设置一个额外的参数,获得传来的字符串:
def getPostSlot(self, top_post):
self.resultList.addItem(top_post)
self.searchProgBar.setValue(self.searchProgBar.value() + 1)
将获得的新闻标题呈现在列表中,并调整进度条的数值。
五、总结
到此为止,我们已经完成所有工作:
- 从新闻网站获取新闻的线程
- 线程与主线程的连接
- 如何实现自定义信号
- 如何使用内建信号
注意:在QThread线程类中处理数据,通过信号将数据发送到主界面线程,进而更新界面元素
看一下现在界面是怎么样的吧:
你将看到:
- 每获得一条新数据,界面立即更新
- 界面仍然可响应,比如拖动、改变输入框内容
- 主线程没有被锁住
- 随时可以点击停止按钮,停止获取数据
That's all.您可以在github上获得程序源码,有任何问题欢迎指出。
Qt——线程类QThread的更多相关文章
- Qt 线程基础(QThread、QtConcurrent等)
[-] 使用线程 何时使用其他技术替代线程 应该使用 Qt 线程的哪种技术 Qt线程基础 QObject与线程 使用互斥量保护数据的完整 使用事件循环防止数据破坏 处理异步执行 昨晚看Qt的Manua ...
- Qt 线程基础(QThread、QtConcurrent等) 2
使用线程 基本上有种使用线程的场合: 通过利用处理器的多个核使处理速度更快. 为保持GUI线程或其他高实时性线程的响应,将耗时的操作或阻塞的调用移到其他线程. 何时使用其他技术替代线程 开发人员使用线 ...
- Qt 线程基础(QThread、QtConcurrent、QThreadPool等)
使用线程 基本上有种使用线程的场合: 通过利用处理器的多个核使处理速度更快. 为保持GUI线程或其他高实时性线程的响应,将耗时的操作或阻塞的调用移到其他线程. 何时使用其他技术替代线程 开发人员使 ...
- Qt线程(2) QThread中使用WorkObject
一般继承QThread的WorkThread都会在重载的run()中创建临时的WorkObject,这样能确定这个WorkObject在该thread中使用 那如果这个WorkObject是个Sing ...
- Qt 学习之路 :Qt 线程相关类
希望上一章有关事件循环的内容还没有把你绕晕.本章将重新回到有关线程的相关内容上面来.在前面的章节我们了解了有关QThread类的简单使用.不过,Qt 提供的有关线程的类可不那么简单,否则的话我们也没必 ...
- Qt 学习之路 2(73):Qt 线程相关类
Home / Qt 学习之路 2 / Qt 学习之路 2(73):Qt 线程相关类 Qt 学习之路 2(73):Qt 线程相关类 豆子 2013年11月26日 Qt 学习之路 2 7条评论 希 ...
- Qt线程(1) moveToThread
若在Qt准备使用线程类一般有两种方式(1) 采用WorkObject配合QThread进行使用 (2)继承QThread, 重载run()函数即可. 注:采用Qt::Concurrent之类的不在本文 ...
- QT核心编程之Qt线程 (c)
QT核心编程之Qt线程是本节要介绍的内容,QT核心编程我们要分几个部分来介绍,想参考更多内容,请看末尾的编辑推荐进行详细阅读,先来看本篇内容. Qt对线程提供了支持,它引入了一些基本与平台无关的线程类 ...
- Qt 线程基础(Thread Basics的翻译,线程的五种使用情况)
Qt 线程基础(QThread.QtConcurrent等) 转载自:http://blog.csdn.net/dbzhang800/article/details/6554104 昨晚看Qt的Man ...
随机推荐
- C#四则运算器(多态方法实现)
在上一节C#课上,我们学习了用类的继承的方式来做一个四则运算器,然而老师的代码在课上演示的效果并不理想,而且没有使用多态的思想实现,今天我们就来用多态的方式实现四则运算器. 1. 题目及要求 2. A ...
- 【SIKIA计划】_11_Unity动画插件-DOTween笔记
[插值移动]using DG.Tweening;public class GetStart:MomoBehaviour{ public Vector3 myValue = new Vector3(0, ...
- Python连接MySQL数据库(pymysql的使用)
本文Python版本3.5.3,mysq版本5.7.23 基本使用 # 导入pymysql模块 import pymysql #连接数据库 conn = pymysql.connect( databa ...
- java基础---JDK、JRE、JVM的区别和联系
当我们学习java语言时,首先需要安装到我们电脑上的就是jdk.jdk是java语言的开发环境,只有安装了jdk,我们才能使用java语言开发程序. JDK=JRE+开发工具包 JRE=JVM+核心类 ...
- C#中字符串 "驻留"与Lock(转载)
class TestWorker 2 { 3 public void DoMultiThreadedWork(object someParameter) 4 { 5 ...
- python-python爬取豆果网(菜谱信息)
#-*- coding = utf-8 -*- #获取豆果网图片 import io from bs4 import BeautifulSoup import requests #爬取菜谱的地址 ur ...
- 简单理解DNS解析流程(一)
0x0 简单理解dns DNS服务器里存着一张表 表中放着域名和IP地址,域名和IP地址以映射关系保存,即一对一 浏览器访问某个域名,实际上是访问它的ip地址 所以浏览器需要知道域名对应的ip地址 如 ...
- jdk8 Optional使用详解
思考: 调用一个方法得到了返回值却不能直接将返回值作为参数去调用别的方法. 原来解决方案: 我们首先要判断这个返回值是否为null,只有在非空的前提下才能将其作为其他方法的参数.这正是一些类似Guav ...
- check the manual that corresponds to your MySQL server version for the right syntax to use near 'desc
往一个新建的MySQL表中插入数据报错 2018-08-22 14:09:18.875 ERROR 9256 --- [apr-8080-exec-9] o.s.b.w.servlet.support ...
- NABC for Teamproject
“教育是一个社会发展的支柱, 你和我能看到并理解这个博客, 教育功不可没. 高等教育的形式并不是一成不变的, 高等教育一直在演进.”邹欣老师在博客上如此写道.为了迎合信息化时代的特色,网络上的知识传 ...