QT+多线程TCP服务器+进阶版

avatar
作者
猴君
阅读量:0

        针对之前的服务器,如果子线程工作类里面需要使用socket发送消息,必须要使用信号与槽的方法, 先发送一个信号给父进程,父进程调用socket发送消息(原因是QT防止父子进程抢夺同一资源,因此直接规定父子进程不能使用同一资源,可能很片面,但至少针对socket这个类是这样的),因此,为了更好的使用多线程TCP服务器,下面介绍一种新的方法。

1、新建一个编译无错误的工程

2、在pro文件中加入network

3、新建两个类,一个Worker类继承QObject,一个TcpServer类继承QTcpServer

4、编辑服务器界面

5、给控件命名

6、查找本地有效的IPV4,并用combox控件显示

    QList<QHostAddress> ipAddressesList = QNetworkInterface::allAddresses();     for (int i = 0; i < ipAddressesList.size(); ++i) {         if (!ipAddressesList.at(i).isNull()//IP地址是否为NULL             && ipAddressesList.at(i).toIPv4Address() != 0//是否是IPV4地址             )         {             ui->IPComboBox->addItem(ipAddressesList.at(i).toString());         }     }

7、创建Server对象,绑定连接服务器按钮和创建代码

#include "tcpserver.h" TcpServer *tcpserver = NULL;//TcpServer这个是自定义的服务器类  void Widget::on_connectPushButton_clicked() {     if(ui->connectPushButton->text().contains("打开服务器"))     {         tcpserver = new TcpServer();           //实例化tcpserver对象         tcpserver->listen( QHostAddress(ui->IPComboBox->currentText()),ui->portLineEdit->text().toInt());//进行IP地址与端口的监听         ui->connectPushButton->setText("断开服务器");     }     else     {         tcpserver->close();         ui->connectPushButton->setText("打开服务器");     } } 

8、当客户端被连接时,TcpServer类会自己调用incomingConnection方法,然后我们需要重写这个方法,首先就是新建一个工作类,然后把套接字文件符传送过去,然后在构造函数初始化套接字

protected:     void incomingConnection(qintptr socketDescriptor) override; //当客户端连接时被调用。
void TcpServer::incomingConnection(qintptr socketDescriptor) {     QThread *thread = new QThread();     Worker *worker = new Worker(socketDescriptor);//socketDescriptor这个参数时套接字的描述符,用于生成套接字socket      connect(thread, &QThread::finished, thread, &QThread::deleteLater); //释放线程资源      worker->moveToThread(thread);     thread->start(); } 

 9、先修改工作类worker的构造函数,将套接字描述符加入

public:     explicit Worker(qintptr socketDescriptor,QObject *parent = nullptr);   Worker::Worker(qintptr socketDescriptor, QObject *parent)     : QObject{parent} {     m_socketDescriptor=socketDescriptor;      m_tcpsocket = new QTcpSocket(this);     m_tcpsocket->setSocketDescriptor(m_socketDescriptor); }

10、获取套接字之后开始处理读到的数据,在工作类中建立读信号和槽函数关系

public slots:     void onReadyRead();   connect(m_tcpsocket,&QTcpSocket::readyRead,this,&Worker::onReadyRead);   void Worker::onReadyRead() {     QByteArray data = m_tcpsocket->readAll();     m_tcpsocket->write("receive : "+data); } 

12、编译通过,且连接成功,可以将收到的消息发给服务器

 13、开始处理发送信息的消息,首先Widget界面有一个发送按钮,用于发送数据给客户端,代码实际操作时发送信号给工作类处理

signals:     void sendDataSignals(QString data);  void Widget::on_sendPushButton_clicked() {     emit sendDataSignals(ui->sendPlainTextEdit->toPlainText()); } 

我们如何让Widget和worker工作类产生联系,利用中间类TcpServer,我们先将sendDataSignals信号发送到TcpServer类中,利用connect信号转信号,然后等待TcpServer发送 sendDataSignals信号时绑定Worker类的槽函数进行发送数据

connect(this,&Widget::sendDataSignals,tcpserver,&TcpServer::sendDataSignals);
connect(this,&TcpServer::sendDataSignals,worker,&Worker::sendData_slots);
void Worker::sendData_slots(QString data) {      m_tcpsocket->write("send : "+data.toUtf8()+"\r\n"); }

 14、编译代码,运行成功,可以成功收发数据

15、当接收到数据后我们让其在网络数据接收窗口显示,采用信号与槽的机制

void Worker::onReadyRead() {     QByteArray data = m_tcpsocket->readAll();     m_tcpsocket->write("i am server,receive : "+data+"\r\n");     emit readDisplaySignals(QString::fromUtf8(data)); }
connect(worker,&Worker::readDisplaySignals,this,&TcpServer::readDisplaySignals);
connect(tcpserver,&TcpServer::readDisplaySignals,this,&Widget::readDataDisplay_slots);
void Widget::readDataDisplay_slots(QString data) {     QString str=ui->receivePlainTextEdit->toPlainText();//接收区之前的内容     QDateTime nowtime = QDateTime::currentDateTime();     str += "[" + nowtime.toString("yyyy-MM-dd hh:mm:ss") + "] "+ "RX: ";//记录当前时间     str = str + data + "\r\n";//当前接收最新消息     ui->receivePlainTextEdit->setPlainText(str); }

16、测试成功

17、完整工程代码

链接:https://pan.baidu.com/s/1wqfFoLKSrARjDHIX0Y-diQ?
提取码:生日

广告一刻

为您即时展示最新活动产品广告消息,让您随时掌握产品活动新动态!