66.QT-线程并发、QTcpServer并发、QThreadPool线程池
1.线程并发
一个程序内部能拥有多个线程并行执行。一个线程的执行可以被认为是一个CPU在执行该程序。
当一个程序运行在多线程下,就好像有多个CPU在同时执行该程序。
总之,多线程即可以这么理解:多线程是处理高并发的一种编程方法,即并发需要用多线程实现。
2.如何分配线程数量
利用 CPU 核心数,应用并发编程来提高效率.线程IO时间所占比例越高,需要越多线程;线程CPU时间所占比例越高,需要越少线程。
理论上:
线程数量 = CPU 核数(逻辑)+ 1
为什么+1,《Java并发编程实战》这么说:
- 计算(CPU)密集型的线程恰好在某时因为发生一个页错误或者因其他原因而暂停,刚好有一个“额外”的线程,可以确保在这种情况下CPU周期不会中断工作。
IO时间和CPU时间
- IO操作实际就是不需要CPU介入,比如DMA请求,比如把内容从硬盘上读到内存的过程,或者是从网络上接收信息到本机内存的过程(sleep也可以算IO操作)
- CPU操作实际就是进行大量的计算,消耗CPU资源,比如计算圆周率、对视频进行高清解码等等,全靠CPU的运算能力。
所以对于单核CPU而言:
最佳线程数 = 1 + (IO操作耗时/CPU操作耗时)
比如: IO操作耗时为500ms、CPU操作耗时为1500ms
最佳线程数 = 1 + (IO操作耗时/CPU操作耗时) = 1 + (500/1500) = 4
对于多核CPU而言:
最佳线程数 = CPU核心数 * (1 + (IO操作耗时/CPU操作耗时))
3.QTcpServer并发
QTcpServer要实现并发,首先需要子类化QTcpServer,然后重写incomingConnection()函数.该函数定义如下所示:
[virtual protected] void QTcpServer::incomingConnection(qintptr socketDescriptor)
// 当有新连接时,首先会调用该函数,通过socketDescriptor参数(连接本机的套接字)创建一个QTcpSocket,设置套接字描述符,然后将QTcpSocket存储在一个内部挂起连接列表中。最后触发newConnection()。
我们重写该函数,通过一个QThread将socketDescriptor参数传到一个线程中,然后调用socketDescriptor()函数初始化一个QTcpSocket.从而达到QThread中生成一个新的QTcpSocket.
MyServer重写如下所示:
void MyServer::incomingConnection(qintptr socketDescriptor)
{
MyThread *thread = new MyThread(socketDescriptor, this);
connect(thread, SIGNAL(finished()), thread, SLOT(deleteLater()));
thread->start();
}
MyThread重写run如下所示:
void MyThread::run()
{
QTcpSocket tcpSocket;
// 初始化一个QTcpSocket
if (!tcpSocket.setSocketDescriptor(socketDescriptor)) {
emit error(tcpSocket.error());
return;
}
// 发送字符串
tcpSocket.write("123456".toLocal8Bit());
tcpSocket.disconnectFromHost();
tcpSocket.waitForDisconnected();
}
然后在widget中:
server.listen(QHostAddress::AnyIPv4,8080);
每当一个client连接该server时,就会接收到"123456",然后被断开.
4.线程池概念
假如服务器突然来了500个任务,但是我们最佳线程数是20个,不可能立马创建500个线程,因为线程过多会带来调度开销,进而影响缓存局部性和整体性能。
所以我们需要线程池,线程池不仅能够保证内核的充分利用,还能防止过分调度。
线程池就相当于排队去银行办理业务.排队的人就是要处理业务的任务线程,客服就是线程池中容纳办理业务的最大数量.每当一个办理业务的线程结束后,线程池就会从等待队列中取出一个线程进行业务办理.
5.QThreadPool并发线程池
在Qt中,线程池可以使用QThreadPool类,用来管理多个QThread的集合.
QThreadPool管理和回收单独的QThread对象,以帮助减少使用线程的程序中创建线程的成本。
每个Qt应用程序都有一个全局QThreadPool对象,可以通过调用globalInstance()来访问(也可以自己定义个QThreadPool)。
要使用一个QThreadPool线程,需要子类化QRunnable.并实现run()虚函数。
然后创建一个子类化QRunnable类的一个对象,并将其传递给QThreadPool::start(),来启动一个线程.start()函数如下所示:
void QThreadPool::start(QRunnable *runnable, int priority = 0)
// 启动一个runnable,如果当前线程池数量超过了maxThreadCount(),那么将runnable添加到等待队列中.
// priority参数可用于控制runnable在等待队列中的被执行的顺序。
// 默认runnable->autoDelete()返回true,线程池将获得可运行对象的所有权,并且在runnable->run()返回后,可运行对象将被线程池自动删除。
// 可以通过QRunnable::setAutoDelete()来更改自动删除标志
QThreadPool支持通过在QRunnable::run()中调用tryStart(this)来多次执行同一个QRunnable。
如果autoDelete被启用,QRunnable将在最后一个线程退出run函数时被删除。
当autoDelete启用时,使用相同的QRunnable多次调用start()会创建一个竞争条件,不建议这样做。
在一定时间内未使用的线程将过期。默认超时时间为30000毫秒(30秒)。这可以使用setExpiryTimeout(int)来更改。设置负数将禁用过期机制。
调用maxThreadCount()查询要使用的最大线程数。也可以使用setMaxThreadCount()来更改这个限制。默认值是QThread::idealThreadCount(). 该函数定义如下所示:
[static] int QThread::idealThreadCount()
//返回系统上可以运行的理想线程数。这是通过查询系统中真实的和逻辑的处理器核的数量来完成的。
//如果无法检测到处理器核数,则该函数返回1。
示例如下所示:
#include <QCoreApplication>
#include <QThread>
#include <QDebug>
#include <QRunnable>
#include <QThreadPool> class ComputeTask : public QRunnable
{
int index;
void run() override
{
const int work = 1000 * 1000 * 40; // 每个任务计数40000000次
volatile int v = 0;
for (int j = 0; j < work; ++j)
++v;
qDebug() << index << " thread: " << QThread::currentThreadId();
} public:
ComputeTask(int i) {
index = i;
} }; int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv); const int cnt = 200; // 200个任务 QThreadPool pool;
qDebug() << "maxThreadCount: " << pool.maxThreadCount();
for (int i = 0; i < cnt; ++i) {
ComputeTask *compute = new ComputeTask(i);
pool.start(compute);
} return a.exec();
}
打印如下所示:
66.QT-线程并发、QTcpServer并发、QThreadPool线程池的更多相关文章
- java线程安全之并发Queue
关闭 原 java线程安全之并发Queue(十三) 2017年11月19日 23:40:23 小彬彬~ 阅读数:12092更多 所属专栏: 线程安全 版权声明:本文为博主原创文章,未经博主允许不 ...
- <关于并发框架>Java原生线程池原理及Guava与之的补充
原创博客,转载请联系博主! 转眼快两个月没有更新自己的博客了. 一来感觉自己要学的东西还是太多,与其花几个小时写下经验分享倒不如多看几点技术书. 二来放眼网上已经有很多成熟的中文文章介绍这些用法,自己 ...
- 【Java并发系列04】线程锁synchronized和Lock和volatile和Condition
img { border: solid 1px } 一.前言 多线程怎么防止竞争资源,即防止对同一资源进行并发操作,那就是使用加锁机制.这是Java并发编程中必须要理解的一个知识点.其实使用起来还是比 ...
- 【java并发编程实战】-----线程基本概念
学习Java并发已经有一个多月了,感觉有些东西学习一会儿了就会忘记,做了一些笔记但是不系统,对于Java并发这么大的"系统",需要自己好好总结.整理才能征服它.希望同仁们一起来学习 ...
- Netty : writeAndFlush的线程安全及并发问题
使用Netty编程时,我们经常会从用户线程,而不是Netty线程池发起write操作,因为我们不能在netty的事件回调中做大量耗时操作.那么问题来了 – 1, writeAndFlush是线程安全的 ...
- C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题
(补充:初始化FileStream时使用包含文件共享属性(System.IO.FileShare)的构造函数比使用自定义线程锁更为安全和高效,更多内容可点击参阅) 在开发程序的过程中,难免少不了写入错 ...
- java核心知识点学习----并发和并行的区别,进程和线程的区别,如何创建线程和线程的四种状态,什么是线程计时器
多线程并发就像是内功,框架都像是外功,内功不足,外功也难得精要. 1.进程和线程的区别 一个程序至少有一个进程,一个进程至少有一个线程. 用工厂来比喻就是,一个工厂可以生产不同种类的产品,操作系统就是 ...
- PAIP.并发编程 多核编程 线程池 ExecutorService的判断线程结束
PAIP.并发编程 多核编程 线程池 ExecutorService的判断线程结束 ExecutorService并没有提供什么 isDone()或者isComplete()之类的方法. 作者Atti ...
- 线程高级应用-心得6-java5线程并发库中同步工具类(synchronizers),新知识大用途
1.新知识普及 2. Semaphore工具类的使用案例 package com.java5.thread.newSkill; import java.util.concurrent.Executor ...
随机推荐
- [c++] 如何流畅地读写代码
代码不同于普通文字,阅读时注意两方面: 符号含义:相同符号,上下文不同时含义也不同,如*和& 阅读顺序:不总是按从左往右顺序阅读的,有时要倒着读或者跳着读逻辑才通顺 适当省略:有些内容虽然写了 ...
- 用 set follow-fork-mode child即可。这是一个 gdb 命令,其目的是告诉 gdb 在目标应用调用fork之后接着调试子进程而不是父进程,因为在 Linux 中fork系统调用成功会返回两次,一次在父进程,一次在子进程
GDB的那些奇淫技巧 evilpan 收录于 Security 2020-09-13 约 5433 字 预计阅读 11 分钟 709 次阅读 gdb也用了好几年了,虽然称不上骨灰级玩家,但 ...
- [笔记] 《c++ primer》书店程序 Chapter2
Sales_data.h 1 #ifndef SALES_DATA_H 2 #define SALES_DATA_H 3 4 #include "Version_test.h" 5 ...
- [刷题] 279 Perfect Squares
要求 给出一个正整数n,寻找最少的完全平方数,使他们的和为n 示例 n = 12 12 = 4 + 4 + 4 输出:3 边界 是否可能无解 思路 贪心:12=9+1+1+1,无法得到最优解 图论:从 ...
- createrepo 建立本地yum源
linux使用createrepo制作本地yum源 目录 linux使用createrepo制作本地yum源 安装createrepo软件包 进入本地rpm包目录 执行完后可以看到生成的repod ...
- 学完了这篇JVM,面试官真拿我没办法了!
在我们面试中经常会遇到面试官问一些有关JVM的问题,下面我大概从运行时数据域.类加载机制.类加载器.垃圾收集器.垃圾收集算法.JVM堆内存模型.JVM内存结构.JVM调优等几个方面来讲一下JVM. 一 ...
- MongoDB学习笔记:MongoDB 数据库的命名、设计规范
MongoDB学习笔记:MongoDB 数据库的命名.设计规范 第一部分,我们先说命名规范. 文档 设计约束 UTF-8 字符 不能包含 \0 字符(空字符),这个字符标识建的结尾 . 和 $ ...
- NVIDIA DGX SUPERPOD 企业解决方案
NVIDIA DGX SUPERPOD 企业解决方案 实现大规模 AI 创新的捷径 NVIDIA DGX SuperPOD 企业解决方案是业界首个支持任何组织大规模实施 AI 的基础架构解决方案.这一 ...
- 深度树匹配模型(TDM)
深度树匹配模型(TDM) 算法介绍 Tree-based Deep Match(TDM)是由阿里妈妈精准定向广告算法团队自主研发,基于深度学习上的大规模(千万级+)推荐系统算法框架.在大规模推荐系统的 ...
- CVPR2020:基于层次折叠的跳跃式注意网络点云完成
CVPR2020:基于层次折叠的跳跃式注意网络点云完成 Point Cloud Completion by Skip-Attention Network With Hierarchical Foldi ...