QTcpSocket 服务端和客户端

前提:

pro文件中添加 QT += network

服务端主要采用信号槽机制,代码如如下

核心代码头文件#ifndef TCPSERVER_H
#define TCPSERVER_H#include <QObject>#include <QTcpServer>
#include <QTcpSocket>
#include <QDebug>
#include <QMutex>
#include <QByteArray>
#include <QQueue>class OneSession{public:QTcpSocket* m_clientSocket;QByteArray m_buffer;QQueue<QByteArray> m_msgQueue;QString m_carName;int m_id;QString m_longitude;QString m_latitude;QString m_altitude;OneSession(){m_id = 0;}~OneSession(){}
};class CenterTcpSvr : public QTcpServer
{Q_OBJECT
public:CenterTcpSvr(QObject *parent = nullptr): QTcpServer(parent){}QList<OneSession> m_sessions;QMutex m_mutexForSessions;
protected:void incomingConnection(qintptr socketDescriptor) override;void readyReadFun();void clientDisconnectedFun();void printAllSessions();int parseHead(OneSession& _one);int parseCmdAndDeal(OneSession& _one);
};#endif // TCPSERVER_H核心代码源文件#include "centertcpsvr.h"void CenterTcpSvr::incomingConnection(qintptr socketDescriptor)
{QTcpSocket *socket = new QTcpSocket(this);if (!socket->setSocketDescriptor(socketDescriptor)) {qDebug() << "Error setting socket descriptor";socket->deleteLater();return;}{ //该括号配合锁使用,请不要随意删除,避免造成锁范围扩大QMutexLocker locker(&m_mutexForSessions); // 使用QMutexLocker自动管理互斥锁OneSession one;one.m_clientSocket = socket;m_sessions.append(one);printAllSessions();}// QTcpSocket::connectedconnect(socket, &QTcpSocket::readyRead, this, &CenterTcpSvr::readyReadFun);connect(socket, &QTcpSocket::disconnected, this, &CenterTcpSvr::clientDisconnectedFun);
}struct ProtocolHead{char flag[4];char version[4];char id[4]; //自己的IDchar cmd[4]; //1,注册;2,发给教辅;其他为车标号目标char bodySize[4];
};struct UserProtocolForward{char dst[4];
};int CenterTcpSvr::parseHead(OneSession& _one)
{int ret = -1;while (_one.m_buffer.size() > sizeof(ProtocolHead)) {char* tmpBuf = _one.m_buffer.data();if (_one.m_id == 0) {unsigned  id =     ((unsigned int)tmpBuf[8]  << 24 & 0xFF000000)+((unsigned int)tmpBuf[9]  << 16 & 0x00FF0000)+((unsigned int)tmpBuf[10] <<  8 & 0x0000FF00)+((unsigned int)tmpBuf[11]       & 0x000000FF);_one.m_id = id;}unsigned  bodySize = ((unsigned int)tmpBuf[16]  << 24 & 0xFF000000)+((unsigned int)tmpBuf[17]  << 16 & 0x00FF0000)+((unsigned int)tmpBuf[18] <<  8 & 0x0000FF00)+((unsigned int)tmpBuf[19]       & 0x000000FF);qDebug() << "bodySize is:" << bodySize;int outSize = sizeof(ProtocolHead) + bodySize;if (_one.m_buffer.size() >= outSize) {_one.m_msgQueue.enqueue(_one.m_buffer.left(outSize));_one.m_buffer.remove(0, outSize);ret = 0;} else {break;}}return  ret;
}int CenterTcpSvr::parseCmdAndDeal(OneSession& _one)
{int ret = 0;while (_one.m_msgQueue.size() > 0) {QByteArray bAarray = _one.m_msgQueue.dequeue();qDebug() << "dequeue content is:" << bAarray;char* tmpBuf = bAarray.data();unsigned  cmd =     ((unsigned int)tmpBuf[12]  << 24 & 0xFF000000)+((unsigned int)tmpBuf[13]  << 16 & 0x00FF0000)+((unsigned int)tmpBuf[14] <<  8 & 0x0000FF00)+((unsigned int)tmpBuf[15]       & 0x000000FF);if (cmd == 1){ //注册} else if (cmd == 2) { //发送给教辅} else {bool isIn = false;QList<OneSession>::iterator it;for(it = m_sessions.begin(); it != m_sessions.end(); ++it) {qDebug() << "cmd is : " << cmd  << "_one.m_id: " << _one.m_id ;if (cmd == it->m_id) { // 进到这里表示发送目标it->m_clientSocket->write(bAarray.data(), bAarray.size());it->m_clientSocket->waitForBytesWritten(1000);}}if (isIn) {ProtocolHead sendDataStruct;QByteArray sd;char body[] = {"{\"code\":0, \"msg\":OK}"};int bsize = sizeof(body);memcpy(sendDataStruct.flag, "#GD#", 4);memcpy(sendDataStruct.version, "0001", 4);sendDataStruct.bodySize[0] = 0 >> 24 & 0x000000FF;sendDataStruct.bodySize[1] = 0 >> 16 & 0x000000FF;sendDataStruct.bodySize[2] = 0 >> 8 & 0x000000FF;sendDataStruct.bodySize[3] = 0 & 0x000000FF;char* csend = new char[sizeof (ProtocolHead)+bsize];memset(csend, 0, sizeof(ProtocolHead) + bsize);memcpy(csend, (char*)(&sendDataStruct), sizeof(ProtocolHead));//_one.m_clientSocket->write(csend, sizeof(ProtocolHead) + bsize);_one.m_clientSocket->waitForBytesWritten(1000);delete []csend;}}}return ret;
}void CenterTcpSvr::readyReadFun() {QTcpSocket *clientSocket = qobject_cast<QTcpSocket*>(sender());//qDebug() << "readyReadData :" << clientSocket;QByteArray data = clientSocket->readAll();qDebug() << "data :" << data;QList<OneSession>::iterator it;for(it = m_sessions.begin(); it != m_sessions.end(); ++it) {if(clientSocket == it->m_clientSocket) {it->m_buffer.append(data);break;}}if (-1 == parseHead(*it)) { //解析头并分包放入队列return;}parseCmdAndDeal(*it);}void CenterTcpSvr::printAllSessions() { //此处没有上锁,请在上层使用处控制qDebug() << "clientDisconnectedFun begin. <-----------";QList<OneSession>::iterator it;for(it = m_sessions.begin(); it != m_sessions.end(); ++it) {qDebug() << "所有会话:::" << it->m_clientSocket;}qDebug() << "clientDisconnectedFun   end. ----------->";
}void CenterTcpSvr::clientDisconnectedFun() {QTcpSocket *clientSocket = qobject_cast<QTcpSocket*>(sender());qDebug() << "clientDisconnected :" << clientSocket;if (clientSocket) {// 移除断开连接的客户端socketQMutexLocker locker(&m_mutexForSessions); // 使用QMutexLocker自动管理互斥锁QList<OneSession>::iterator it;for(it = m_sessions.begin(); it != m_sessions.end(); ++it) {//qDebug() << "所有会话:" << clientSocket;if (it->m_clientSocket == clientSocket){clientSocket->deleteLater();m_sessions.erase(it);qDebug() << "删除会话:" << clientSocket;break;}}printAllSessions();}
}调用上述代码头文件#ifndef TEST_H
#define TEST_H#include <QObject>#include <QCoreApplication>
#include <QNetworkAccessManager>
#include <QNetworkRequest>
#include <QNetworkReply>
#include <QUrl>
#include <QDebug>
#include <QThread>
#include <QString>
#include "centertcpsvr.h"class ThreadForCenterTcpSvr : public QThread
{Q_OBJECT
public:ThreadForCenterTcpSvr();void run(); //线程任务函数signals:void startsignal();  //启动⼦线程的信号
};#endif // TEST_H调用上述代码源文件#include "threadforcentertcpsvr.h"
#include <QCoreApplication>
#include "centertcpsvr.h"
#include <QByteArray>
#include <QCoreApplication>
#include <QNetworkAccessManager>
#include <QNetworkRequest>
#include <QNetworkReply>
#include <QUrl>
#include <QDebug>
#include <QThread>
#include <QString>ThreadForCenterTcpSvr::ThreadForCenterTcpSvr()
{}void ThreadForCenterTcpSvr::run()
{while (true){CenterTcpSvr* server = new CenterTcpSvr();if (!server->listen(QHostAddress::Any, 8080)) {qDebug() << "Server failed to start";return;}qDebug() << "Server start";QEventLoop loop;loop.exec(); // 进入事件循环,直到请求完成//QThread::exec(); //或者用这个也可以}
}在 main 函数中启动服务int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);ThreadForCenterTcpSvr* p = new ThreadForCenterTcpSvr();p->start();return a.exec();
}

客户端有两种实现方式

第一种方式如下,非信号槽方式实现

#include <QCoreApplication>#include <QTcpSocket>
#include <QThread>
#include <QByteArray>
#include <QAbstractSocket>struct ProtocolHead{char flag[4];char version[4];char id[4]; //自己的IDchar cmd[4]; //0,注册;1,发给教辅;其他为车标号目标char bodySize[4];
};int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);QTcpSocket socket;while (true) {// 连接到服务器socket.connectToHost("localhost", 8080);// 等待连接if(socket.waitForConnected(100000)) {qDebug() << "Connected!";int c = 0;while (true) {// 发送数据ProtocolHead sendDataStruct;QByteArray sd;char body[] = "Hello server!";int bsize = sizeof(body);memcpy(sendDataStruct.flag, "#GD#", 4);memcpy(sendDataStruct.version, "0001", 4);int id = 48102; //自己的标识sendDataStruct.id[0] = id >> 24 & 0x000000FF;sendDataStruct.id[1] = id >> 16 & 0x000000FF;sendDataStruct.id[2] = id >> 8 & 0x000000FF;sendDataStruct.id[3] = id & 0x000000FF;int cmd = 48102; //对方的标识sendDataStruct.cmd[0] = cmd >> 24 & 0x000000FF;sendDataStruct.cmd[1] = cmd >> 16 & 0x000000FF;sendDataStruct.cmd[2] = cmd >> 8 & 0x000000FF;sendDataStruct.cmd[3] = cmd & 0x000000FF;sendDataStruct.bodySize[0] = bsize >> 24 & 0x000000FF;sendDataStruct.bodySize[1] = bsize >> 16 & 0x000000FF;sendDataStruct.bodySize[2] = bsize >> 8 & 0x000000FF;sendDataStruct.bodySize[3] = bsize & 0x000000FF;char* csend = new char[sizeof (ProtocolHead)+bsize];memset(csend, 0, sizeof(ProtocolHead) + bsize);int pos = 0;memcpy(csend+pos, (char*)(&sendDataStruct), sizeof(ProtocolHead));pos += sizeof(ProtocolHead);memcpy(csend+pos, body, bsize);socket.write(csend, sizeof (ProtocolHead)+bsize);// 等待服务器响应if(socket.waitForBytesWritten(1000)){if (c%250000==0) {qDebug() << "Message sent!";}// 等待服务器的响应while(!socket.waitForReadyRead(1000)) {//QByteArray qb = socket.readAll();if (socket.error() == QAbstractSocket::SocketTimeoutError) {qDebug() << "超时错误:没有在指定时间内接收到数据";}if (socket.error() == QAbstractSocket::RemoteHostClosedError) {qDebug() << "远程主机关闭。";while (true) {//socket.abort();socket.connectToHost("localhost", 8080);// 等待连接if(socket.waitForConnected(1000)) {qDebug() << "Connected!";break;}}} else {qDebug() << "其他错误:" << socket.errorString();}}QByteArray qb = socket.readAll();qDebug() << c << ", ======Response received:" << qb;} else {qDebug() << "Failed to send message";}//break;c++;if (c/250000==10) {c=0;}}// 关闭连接socket.close();} else {qDebug() << "Failed to connect";}QThread::sleep(1);//break;}return a.exec();
}

第二种方式,信号槽方式

文件1#ifndef TCPCLIENT_H
#define TCPCLIENT_H#include <QObject>
#include <QTcpSocket>
#include <QObject>
#include <QTcpSocket>
#include <QThread>
#include <QByteArray>
#include <QAbstractSocket>class TcpClient : public QObject
{Q_OBJECT
public:explicit TcpClient(QObject *parent = nullptr);~TcpClient();void connectToServer(const QString &host, quint16 port);private slots:void onConnected();void onDisconnected();void onReadyRead();void onBytesWritten(qint64 bytes);private:QTcpSocket *socket;
signals:};#endif // TCPCLIENT_H文件2#include "tcpclient.h"TcpClient::TcpClient(QObject *parent) : QObject(parent), socket(new QTcpSocket(this))
{connect(socket, &QTcpSocket::connected, this, &TcpClient::onConnected);connect(socket, &QTcpSocket::disconnected, this, &TcpClient::onDisconnected);connect(socket, &QTcpSocket::readyRead, this, &TcpClient::onReadyRead);connect(socket, &QTcpSocket::bytesWritten, this, &TcpClient::onBytesWritten);
}TcpClient::~TcpClient()
{delete socket; // Qt 自动管理内存,这里只是示例
}void TcpClient::connectToServer(const QString &host, quint16 port)
{socket->connectToHost(host, port);
}void TcpClient::onConnected()
{qDebug() << "Connected to server";socket->write("Hello, Server!");
}void TcpClient::onDisconnected()
{qDebug() << "Disconnected from server";
}void TcpClient::onReadyRead()
{while (socket->canReadLine()){QString line = socket->readLine().trimmed();qDebug() << "Received from server:" << line;}
}void TcpClient::onBytesWritten(qint64 bytes)
{qDebug() << "Sent" << bytes << "bytes to server";
}文件3#include <QCoreApplication>#include <QTcpSocket>
#include <QObject>
#include <QTcpSocket>
#include <QThread>
#include <QByteArray>
#include <QAbstractSocket>
#include "tcpclient.h"int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);TcpClient client;client.connectToServer("127.0.0.1", 8080); // 替换为实际的服务器 IP 和端口return a.exec();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/471727.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue 组件通信及进阶语法

文章目录 一、scoped 样式冲突二、data 是一个函数三、组件通信1. 父子通信1.1 props 校验1.2 props 比较 data 2. 非父子通信2.1 event bus2.2 provide-inject 四、进阶语法1. v-model 详解2. sync 修饰符3. ref 和 $refs4. $nextTick 一、scoped 样式冲突 注意点&#xff1a;…

LeetCode105.从前序与中序遍历构造二叉树

题目要求 给定两个整数数组 preorder 和 inorder &#xff0c;其中 preorder 是二叉树的先序遍历&#xff0c; inorder 是同一棵树的中序遍历&#xff0c;请构造二叉树并返回其根节点。 提示: 1 < preorder.length < 3000inorder.length preorder.length-3000 < pr…

【问卷调研】HarmonyOS SDK开发者社区用户需求有奖调研

问卷请点击&#xff1a;HarmonyOS SDK开发者社区用户需求有奖调研

IOT物联网低代码可视化大屏解决方案汇总

目录 参考来源云服务商阿里云物联网平台产品主页产品文档 开源项目DGIOT | 轻量级工业物联网开源平台项目特点项目地址开源许可 IoTGateway | 基于.NET6的跨平台工业物联网网关项目特点项目地址开源许可 IoTSharp | 基于.Net Core开源的物联网基础平台项目特点项目地址开源许可…

如何在Mac上切换到JDK 17开发环境

在本文中&#xff0c;我将为您介绍如何在Mac上切换到JDK 17&#xff0c;包括下载和安装JDK 17、设置环境变量、在IntelliJ IDEA中配置项目、修改Maven编译配置&#xff0c;并最终使用mvn clean install重新编译项目。通过这个流程&#xff0c;您可以顺利地将开发环境升级到JDK …

玩转ChatGPT:文献阅读 v2.0

一、写在前面 好久不更新咯。 因为最近ChatGPT更新了不少功能&#xff08;水一篇刷存在感&#xff09;&#xff1a; 上线ChatGPT-4o模型&#xff0c;说推理能力还不错&#xff1b;上线联网功能&#xff0c;类似Kimi那种。 所以呢&#xff0c;用它来读文献就挺舒服的了。例如…

自动驾驶系列—从数据采集到存储:解密自动驾驶传感器数据采集盒子的关键技术

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

HarmonyOS本地存储-Preferences(用户首选项)的使用

一&#xff0c;用户首选项简述 ohos.data.preferences (用户首选项) 用户首选项为应用提供Key-Value键值型的数据处理能力&#xff0c;支持应用持久化轻量级数据&#xff0c;并对其修改和查询。 数据存储形式为键值对&#xff0c;键的类型为字符串型&#xff0c;值的存储数据…

SpringCloud 微服务消息队列灰度方案 (RocketMQ 4.x)

目录 背景遇到的问题 RocketMQ 基础基础消息模型扩展后的消息模型部署模型相关概念点 方案对比影子Topic的方案Tag的方案UserProperty的方案影子Group的方案灰度分区的方案方案对比 灰度分区方案设计适配只有部分灰度的情况所做的功能扩展消费者&#xff08;无灰度&#xff09;…

「QT」文件类 之 QDataStream 数据流类

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「QT」QT5程序设计&#x1f4da;全部专栏「Win」Windows程序设计「IDE」集成开发环境「UG/NX」BlockUI集合「C/C」C/C程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「UG/NX」NX定制…

QT<30> Qt中使鼠标变为转圈忙状态

前言&#xff1a;当我们在写软件时&#xff0c;在等待阻塞耗时操作时可以将鼠标变为忙状态&#xff0c;并在一段时间后恢复状态&#xff0c;可以用到GxtWaitCursor&#xff1a;Qt下基于RAII的鼠标等待光标类。 一、效果演示 二、详细代码 在项目中添加C文件&#xff0c;命名为…

Vue的基础使用

一、为什么要学习Vue 1.前端必备技能 2.岗位多&#xff0c;绝大互联网公司都在使用Vue 3.提高开发效率 4.高薪必备技能&#xff08;Vue2Vue3&#xff09; 二、什么是Vue 概念&#xff1a;Vue (读音 /vjuː/&#xff0c;类似于 view) 是一套 构建用户界面 的 渐进式 框架…

数据结构Python版

2.3.3 双链表 双链表和链表一样&#xff0c;只不过每个节点有两个链接——一个指向后一个节点&#xff0c;一个指向前一个节点。此外&#xff0c;除了第一个节点&#xff0c;双链表还需要记录最后一个节点。 每个结点为DLinkNode类对象&#xff0c;包括存储元素的列表data、…

FBX福币交易所恒指收跌1.96% 半导体股继续回调

查查配分析11月14日电 周四,港股三大指数集体下跌。截至收盘,恒生指数跌1.96%,恒生科技指数跌3.08%,恒生中国企业指数跌2.21%。大市成交额1733亿港元。 FBX福币凭借用户友好的界面和对透明度的承诺,迅速在加密货币市场中崭露头角,成为广大用户信赖的平台。 来源:Wind 盘面上,科…

【学习日记】notebook添加JAVA支持

作者是个大学生 这个专栏主要收集课时常用的软件 以及女朋友上课用的软件的教程 新开了gitcode 用于上传安装包 环境说明 windows11 java23.0.1 ijava1.1.2 Anaconda-2024.02 需提前配置好java环境 本篇仅对添加支持进行说明 ijava的GitCode链接NotebookAddsSupportForJava:no…

RabbitMQ 篇-深入了解延迟消息、MQ 可靠性(生产者可靠性、MQ 可靠性、消费者可靠性)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 RabbitMQ 的可靠性 2.0 发送者的可靠性 2.1 生产者重试机制 2.2 生产者确认机制 2.2.1 开启生产者确认机制 2.2.2 定义 ReturnCallback 机制 2.2.3 定义 ConfirmC…

【数据结构】AVL树

引言&#xff1a;在实际情况中&#xff0c;数据不仅仅要存储起来&#xff0c;还要进行对数据进行搜索&#xff0c;为了方便进行高效搜索(在此之前的数据结构的搜索基本都是暴力搜索)二叉搜索树应运而生。但是在极端情况下(我们按照有序的方式进行插入)&#xff0c;二叉搜索树就…

CSS的综合应用例子(网页制作)

这是html的一些最基本内容的代码&#xff1a; <!DOCTYPE html> <html lang"zh"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <t…

MySQL查询某个数据库中特定表的空间占用大小

如果您也想要查询某个数据库中特定表的空间占用大小&#xff0c;包括数据和索引的大小&#xff0c;那么您可以使用以下SQL查询。这个查询将显示特定表在数据库中的数据大小、索引大小以及总大小。 SELECT table_name AS Table,ROUND(((data_length index_length) / 1024 / 10…

Towards Reasoning in Large Language Models: A Survey

文章目录 题目摘要引言什么是推理?走向大型语言模型中的推理测量大型语言模型中的推理发现与启示反思、讨论和未来方向 为什么要推理?结论题目 大型语言模型中的推理:一项调查 论文地址:https://arxiv.org/abs/2212.10403 项目地址: https://github.com/jeffhj/LM-reason…