针对之前的服务器,如果子线程工作类里面需要使用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?
提取码:生日