大数据最新Qt5开发从入门到精通——第十二篇三节(Qt5 事件处理及实例—,2024年最新高并发系统基础篇

img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化资料的朋友,可以戳这里获取

/*此槽函数用于界面上显示的请求次数*/
void Dialog::slotShow()
{
Label2->setText(tr(" 第 %1 次请求完毕。") .arg(++count));
}


#### main.cpp



#include “dialog.h”
#include

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
Dialog w;
w.show();

return a.exec();

}


#### timeserver.cpp



#include “timeserver.h”
#include “timethread.h”
#include “dialog.h”
TimeServer::TimeServer(QObject *parent):QTcpServer(parent)
{
dig =(Dialog *)parent;
}
void TimeServer::incomingConnection(qintptr socketDescriptor)
{
/*以返回的套接字描述符socketDescriptor 创建一个工作线程 TimeThread 。*/
TimeThread *thread= new TimeThread(socketDescriptor,0);

/\*将上述创建的线程结束消息函数 finished()关联到槽函数 slotShow()用于显示请求计数 。 

* 此操作中,因为信号是跨线程的,所以使用了排队连接方式 。
*/
connect (thread, SIGNAL (finished()), dig, SLOT (slotShow ()));

/\*将上述创建的线程结束消息函数 finished()关联到线程自身的槽函数 deleteLater()用于结束线程 。

* 在此操作中,因为信号是在同 一个线程中的,使用了直接连接方式,故最后一个参数可以省略而使用 Qt 的自动连接选择方式 。
* 另外,由于工作线程中存在网络事件,所以不能被外界线程销毁,这里使用了延迟销毁函数 deleteLater()保证由工作线程自身销毁 。
*/
connect (thread, SIGNAL (finished()) , thread, SLOT(deleteLater()),Qt::DirectConnection);
/*启动上述创建的线程 。 执行此语句后,工作线程 (TimeThread) 的虚函数 run()开始执行。*/
thread->start();
}


#### timethread.cpp



#include “timethread.h”
#include
#include
#include
TimeThread::TimeThread(qintptr socketDescriptor,QObject *parent)
:QThread(parent),socketDescriptor(socketDescriptor)
{

}
void TimeThread::run()
{
QTcpSocket tcpSocket; //创建一个 QTcpSocket 类
/*将以上创建的 QTcpSocket 类置以从构造函数中传入的套接字描述符,用于向客户端传回服务器端的当前时间。*/
if(!tcpSocket.setSocketDescriptor(socketDescriptor))
{
/*如果出错,则发出 error(tcpSocket.error())信号报告错误。*/
emit error (tcpSocket.error ());
return;
}
QByteArray block;
QDataStream out(&block,QIODevice::WriteOnly);
out.setVersion (QDataStream::Qt_5_11);
/*如果不出错,则开始获取当前时间。*/

/\*此处需要注意的是时间数据的传输格式, Qt 虽然可以很方便地通过 QDateTime 类的静态函

数 currentDateTime()获取一个时间对象,但类结构是无法直接在网络间传输的,此时需要将它转
换为一个标准的数据类型后再传输 。 而 QDateTime 类提供了 uint toTime_t() const 函数,这个函
数返回当前自 1970-01-01 00:00:00 (UNIX 纪元)经过了多少秒,返回值为一个 uint 类型,可以
将这个值传输给客户端。在客户端方面,使用 QDateTime 类的 void setTime_t(uint seconds)将这
个时间还原 。*/

uint time2u = QDateTime::currentDateTime().toTime\_t(); //(c)
out<<time2u;
tcpSocket.write(block);     //将获得的当前时间传回客户端
tcpSocket.disconnectFromHost(); //断开连接
tcpSocket.waitForDisconnected(); //等待返回

}


## 二、客户端编程


首先在TimeServer.pro中加入 QT +=network



QT += network


### 2.1、效果实例


**图二**


![在这里插入图片描述](https://img-blog.csdnimg.cn/4eba8417590d46e5bdb502847f07b8a2.jpeg#pic_center)


**图三**  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/708c19ab11aa48409b266dd0dbca9fa5.gif#pic_center)


### 2.2、原码实例


#### timeclient.h



#ifndef TIMECLIENT_H
#define TIMECLIENT_H

#include
#include
#include
#include
#include
#include
#include
class TimeClient : public QDialog
{
Q_OBJECT

public:
TimeClient(QWidget *parent = 0);
~TimeClient();
public slots:
void enableGetBtn();
void getTime();
void readTime();
void showError(QAbstractSocket::SocketError socketError);
private:
QLabel *serverNameLabel;
QLineEdit *serverNameLineEdit;
QLabel *portLabel;
QLineEdit *portLineEdit;
QDateTimeEdit *dateTimeEdit;
QLabel *stateLabel;
QPushButton *getBtn;
QPushButton *quitBtn;
uint time2u;
QTcpSocket *tcpSocket;
};

#endif // TIMECLIENT_H


#### main.cpp



#include “timeclient.h”
#include

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
TimeClient w;
w.show();

return a.exec();

}


#### timeclient.cpp



#include “timeclient.h”
#include
#include
#include
#include
#include
TimeClient::TimeClient(QWidget *parent)
: QDialog(parent)
{

/\*初始化界面\*/
setWindowTitle(tr(" 多线程时间服务客户端"));
serverNameLabel =new QLabel(tr("服务器名:"));
serverNameLineEdit = new QLineEdit("Localhost");

portLabel =new QLabel(tr(" 端口:"));
portLineEdit = new QLineEdit;
QGridLayout \*layout= new QGridLayout;
layout->addWidget(serverNameLabel,0,0);
layout->addWidget(serverNameLineEdit,0,1);
layout->addWidget(portLabel,1,0);
layout->addWidget(portLineEdit,1,1);
dateTimeEdit = new QDateTimeEdit(this);
QHBoxLayout \*layout1 = new QHBoxLayout;
layout1->addWidget(dateTimeEdit);
stateLabel =new QLabel(tr("请首先运行时间服务器"));
QHBoxLayout \*layout2 = new QHBoxLayout;
layout2->addWidget(stateLabel);
getBtn = new QPushButton(tr(" 获取时间")) ;
getBtn->setDefault(true);
getBtn->setEnabled(false);
quitBtn = new QPushButton(tr(" 退出")) ;
QHBoxLayout \*layout3 = new QHBoxLayout;
layout3->addStretch ();
layout3->addWidget(getBtn);
layout3->addWidget(quitBtn);
QVBoxLayout \*mainLayout = new QVBoxLayout(this);
mainLayout->addLayout(layout);
mainLayout->addLayout(layout1);
mainLayout->addLayout(layout2);
mainLayout->addLayout(layout3);
connect(serverNameLineEdit, SIGNAL(textChanged(QString)),this, SLOT (enableGetBtn ()));
connect(portLineEdit ,SIGNAL(textChanged(QString)),
this, SLOT (enableGetBtn ()));
connect (getBtn, SIGNAL (clicked()), this, SLOT (getTime()));
connect (quitBtn, SIGNAL (clicked()), this, SLOT (close()));
tcpSocket = new QTcpSocket(this);
connect (tcpSocket, SIGNAL (readyRead ()) , this, SLOT (readTime())) ;
connect (tcpSocket,SIGNAL(error(QAbstractSocket::SocketError)),this,SLOT(showError(QAbstractSocket::SocketError)));
portLineEdit->setFocus () ;

}

TimeClient::~TimeClient()
{

}

/*检查文本框填充的数据是否正常*/
void TimeClient:: enableGetBtn()
{
getBtn->setEnabled(!serverNameLineEdit->text().isEmpty()&&!portLineEdit->text().isEmpty());
}
/*连接到服务端*/
void TimeClient::getTime()
{
getBtn->setEnabled(false);
time2u =0;
/*QTcpServer释放前需要先close。QTcpSocket释放前需要先abort。*/
tcpSocket->abort ();
tcpSocket->connectToHost(serverNameLineEdit->text(),portLineEdit->text ().toInt());
}

/*从服务端接收到数据时刷新时间*/
void TimeClient::readTime()
{
QDataStream in(tcpSocket);
in.setVersion(QDataStream::Qt_5_11);
if(time2u==0)
{
if(tcpSocket->bytesAvailable ()<(int)sizeof(uint))
return;
in>>time2u;
}
dateTimeEdit->setDateTime(QDateTime::fromTime_t(time2u));
getBtn->setEnabled(true);

}

/*当TCP连接异常时,进行报错*/
void TimeClient::showError(QAbstractSocket::SocketError socketError)
{
switch(socketError)
{
case QAbstractSocket::RemoteHostClosedError:
break;
case QAbstractSocket::HostNotFoundError:
QMessageBox::information(this, tr(" 时间服务客户端"),tr(" 主机不可达!“));
break;
case QAbstractSocket::ConnectionRefusedError:
QMessageBox::information(this, tr(” 时间服务客户端"),tr(" 连接被拒绝!“));
break;
default:
QMessageBox::information(this, tr(” 时间服务客户端"),tr(" 产生如下错误:%1.") .arg(tcpSocket->errorString()));
}
getBtn->setEnabled(true);
}


## 三、多线程优势与特性


通常情况下,应用程序都是在一个线程中执行操作。但是,当调用一个耗时操作(例如,大批量 I/0 或大量矩阵变换等 CPU 密集操作)时,用户界面常常会冻结。而使用多线程可解决这一问题。  
 **多线程具有以下优势**。  
 (1) 提高应用程序的响应速度。这对千开发图形界面的程序尤为重要,当一个操作耗时很  
 长时,整个系统都会等待这个操作,程序就不能响应键盘、鼠标、菜单等的操作,而使用多线程技术可将耗时长的操作置于一个新的线程,从而避免出现以上问题。  
 (2) 使多 CPU 系统更加有效。当线程数不大于 CPU 数目时,操作系统可以调度不同的线  
 程运行千不同的 CPU 上。  
 (3) 改善程序结构。一个既长又复杂的进程可以考虑分为多个线程,成为独立或半独立的  
 运行部分,这样有利于代码的理解和维护。  
 **多线程程序具有以下特点。**  
 (1) 多线程程序的行为无法预期,当多次执行上述程序时,每次的运行结果都可能不同。  
 (2) 多线程的执行顺序无法保证,它与操作系统的调度策略和线程优先级等因素有关。  
 (3) 多线程的切换可能发生在任何时刻、任何地点。  
 (4) 由于多线程对代码的敏感度高,因此对代码的细微修改都可能产生意想不到的结果。  
 基于以上这些特点,为了有效地使用线程,开发人员必须对其进行控制。


## 四、总结



![img](https://img-blog.csdnimg.cn/img_convert/973d8e1d465ba2984b64e657e1d5797a.png)
![img](https://img-blog.csdnimg.cn/img_convert/4166a9507c6b465c4962769480818600.png)
![img](https://img-blog.csdnimg.cn/img_convert/39d4e124859eef6a1f38a5180bb6c8fb.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**

行控制。


## 四、总结



[外链图片转存中...(img-ghSriDVy-1715768555780)]
[外链图片转存中...(img-OYQuur6R-1715768555781)]
[外链图片转存中...(img-SeYyD07c-1715768555781)]

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**

  • 5
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值