Qt是C++的一个框架,并且提供了用于套接字通信的类(TCP,UDP)。
使用 Qt 提供的类进行基于 TCP 的套接字通信需要用到两个类:

QTcpServer:服务器类,用于监听客户端连接以及和客户端建立连接。
QTcpSocket:通信的套接字类,客户端、服务器端都需要使用。
这两个套接字通信类都属于网络模块 network。

作者: 苏丙榅
链接: https://www.subingwen.cn/qt/socket-tcp/
来源: 爱编程的大丙

1. QTcpServer

QTcpServer 类用于监听客户端连接以及和客户端建立连接,在使用之前先介绍一下这个类提供的一些常用 API 函数:

1.1 公共成员函数
构造函数

QTcpServer::QTcpServer(QObject *parent = Q_NULLPTR);

给监听的套接字设置监听

bool QTcpServer::listen(const QHostAddress &address = QHostAddress::Any, quint16 port = 0);

// 判断当前对象是否在监听, 是返回true,没有监听返回false

bool QTcpServer::isListening() const;

// 如果当前对象正在监听返回监听的服务器地址信息, 否则返回 QHostAddress::Null

QHostAddress QTcpServer::serverAddress() const;

// 如果服务器正在侦听连接,则返回服务器的端口; 否则返回0

quint16 QTcpServer::serverPort() const

参数:
address:通过类 QHostAddress 可以封装 IPv4、IPv6 格式的 IP 地址,QHostAddress::Any 表示自动绑定
port:如果指定为 0 表示随机绑定一个可用端口。
返回值:绑定成功返回 true,失败返回 false
得到和客户端建立连接之后用于通信的 QTcpSocket 套接字对象,它是 QTcpServer 的一个子对象,当 QTcpServer 对象析构的时候会自动析构这个子对象,当然也可自己手动析构,建议用完之后自己手动析构这个通信的 QTcpSocket 对象。

QTcpSocket *QTcpServer::nextPendingConnection();

阻塞等待客户端发起的连接请求,不推荐在单线程程序中使用,建议使用非阻塞方式处理新连接,即使用信号 newConnection() 。

bool QTcpServer::waitForNewConnection(int msec = 0, bool *timedOut = Q_NULLPTR);

参数:
msec:指定阻塞的最大时长,单位为毫秒(ms)
timeout:传出参数,如果操作超时 timeout 为 true,没有超时 timeout 为 false

1.2 信号

当接受新连接导致错误时,将发射如下信号。socketError 参数描述了发生的错误相关的信息。

[signal] void QTcpServer::acceptError(QAbstractSocket::SocketError socketError);

每次有新连接可用时都会发出 newConnection () 信号。

[signal] void QTcpServer::newConnection();

2. QTcpSocket

QTcpSocket 是一个套接字通信类,不管是客户端还是服务器端都需要使用。在 Qt 中发送和接收数据也属于 IO 操作(网络 IO),先来看一下这个类的继承关系:

2.1 公共成员函数

构造函数

QTcpSocket::QTcpSocket(QObject *parent = Q_NULLPTR);

连接服务器,需要指定服务器端绑定的IP和端口信息。

[virtual] void QAbstractSocket::connectToHost(const QString &hostName, quint16 port, OpenMode openMode = ReadWrite, NetworkLayerProtocol protocol = AnyIPProtocol);[virtual] void QAbstractSocket::connectToHost(const QHostAddress &address, quint16 port, OpenMode openMode = ReadWrite);

在 Qt 中不管调用读操作函数接收数据,还是调用写函数发送数据,操作的对象都是本地的由 Qt 框架维护的一块内存。因此,调用了发送函数数据不一定会马上被发送到网络中,调用了接收函数也不是直接从网络中接收数据,关于底层的相关操作是不需要使用者来维护的。

接收数据

// 指定可接收的最大字节数 maxSize 的数据到指针 data 指向的内存中
qint64 QIODevice::read(char *data, qint64 maxSize);
// 指定可接收的最大字节数 maxSize,返回接收的字符串
QByteArray QIODevice::read(qint64 maxSize);
// 将当前可用操作数据全部读出,通过返回值返回读出的字符串
QByteArray QIODevice::readAll();

发送数据

// 发送指针 data 指向的内存中的 maxSize 个字节的数据
qint64 QIODevice::write(const char *data, qint64 maxSize);
// 发送指针 data 指向的内存中的数据,字符串以 \0 作为结束标记
qint64 QIODevice::write(const char *data);
// 发送参数指定的字符串
qint64 QIODevice::write(const QByteArray &byteArray);

2.2 信号

在使用 QTcpSocket 进行套接字通信的过程中,如果该类对象发射出 readyRead() 信号,说明对端发送的数据达到了,之后就可以调用 read 函数接收数据了。

[signal] void QIODevice::readyRead();

调用 connectToHost() 函数并成功建立连接之后发出 connected() 信号。

[signal] void QAbstractSocket::connected();

在套接字断开连接时发出 disconnected() 信号。

[signal] void QAbstractSocket::disconnected();

3. 通信流程

3.1 服务器端

3.1.1 通信流程

创建套接字服务器 QTcpServer 对象
通过 QTcpServer 对象设置监听,即:QTcpServer::listen()
基于 QTcpServer::newConnection() 信号检测是否有新的客户端连接
如果有新的客户端连接调用 QTcpSocket *QTcpServer::nextPendingConnection() 得到通信的套接字对象
使用通信的套接字对象 QTcpSocket 和客户端进行通信

3.1.2 代码片段

服务器端的窗口界面如下图所示:

头文件

class MainWindow : public QMainWindow
{Q_OBJECTpublic:explicit MainWindow(QWidget *parent = 0);~MainWindow();private slots:void on_startServer_clicked();void on_sendMsg_clicked();private:Ui::MainWindow *ui;QTcpServer* m_server;QTcpSocket* m_tcp;
};

源文件

MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);setWindowTitle("TCP - 服务器");// 创建 QTcpServer 对象m_server = new QTcpServer(this);// 检测是否有新的客户端连接connect(m_server, &QTcpServer::newConnection, this, [=](){m_tcp = m_server->nextPendingConnection();ui->record->append("成功和客户端建立了新的连接...");m_status->setPixmap(QPixmap(":/connect.png").scaled(20, 20));// 检测是否有客户端数据connect(m_tcp, &QTcpSocket::readyRead, this, [=](){// 接收数据QString recvMsg = m_tcp->readAll();ui->record->append("客户端Say: " + recvMsg);});// 客户端断开了连接connect(m_tcp, &QTcpSocket::disconnected, this, [=](){ui->record->append("客户端已经断开了连接...");m_tcp->deleteLater();m_status->setPixmap(QPixmap(":/disconnect.png").scaled(20, 20));});});
}MainWindow::~MainWindow()
{delete ui;
}// 启动服务器端的服务按钮
void MainWindow::on_startServer_clicked()
{unsigned short port = ui->port->text().toInt();// 设置服务器监听m_server->listen(QHostAddress::Any, port);ui->startServer->setEnabled(false);
}// 点击发送数据按钮
void MainWindow::on_sendMsg_clicked()
{QString sendMsg = ui->msg->toPlainText();m_tcp->write(sendMsg.toUtf8());ui->record->append("服务器Say: " + sendMsg);ui->msg->clear();
}

3.2 客户端

客户端的窗口界面如下图所示:

头文件

class MainWindow : public QMainWindow
{Q_OBJECTpublic:explicit MainWindow(QWidget *parent = 0);~MainWindow();private slots:void on_connectServer_clicked();void on_sendMsg_clicked();void on_disconnect_clicked();private:Ui::MainWindow *ui;QTcpSocket* m_tcp;
};

源文件

MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);setWindowTitle("TCP - 客户端");// 创建通信的套接字对象m_tcp = new QTcpSocket(this);// 检测服务器是否回复了数据connect(m_tcp, &QTcpSocket::readyRead, [=](){// 接收服务器发送的数据QByteArray recvMsg = m_tcp->readAll();ui->record->append("服务器Say: " + recvMsg);});// 检测是否和服务器是否连接成功了connect(m_tcp, &QTcpSocket::connected, this, [=](){ui->record->append("恭喜, 连接服务器成功!!!");m_status->setPixmap(QPixmap(":/connect.png").scaled(20, 20));});// 检测服务器是否和客户端断开了连接connect(m_tcp, &QTcpSocket::disconnected, this, [=](){ui->record->append("服务器已经断开了连接, ...");ui->connectServer->setEnabled(true);ui->disconnect->setEnabled(false);});
}MainWindow::~MainWindow()
{delete ui;
}// 连接服务器按钮按下之后的处理动作
void MainWindow::on_connectServer_clicked()
{QString ip = ui->ip->text();unsigned short port = ui->port->text().toInt();// 连接服务器m_tcp->connectToHost(QHostAddress(ip), port);ui->connectServer->setEnabled(false);ui->disconnect->setEnabled(true);
}// 发送数据按钮按下之后的处理动作
void MainWindow::on_sendMsg_clicked()
{QString sendMsg = ui->msg->toPlainText();m_tcp->write(sendMsg.toUtf8());ui->record->append("客户端Say: " + sendMsg);ui->msg->clear();
}// 断开连接按钮被按下之后的处理动作
void MainWindow::on_disconnect_clicked()
{m_tcp->close();ui->connectServer->setEnabled(true);ui->disconnect->setEnabled(false);
}

作者: 苏丙榅
链接: https://www.subingwen.cn/qt/socket-tcp/
来源: 爱编程的大丙

基于 TCP 的 Qt 网络通信相关推荐

  1. Java学习系列(十六)Java面向对象之基于TCP协议的网络通信

    TCP/IP的网络分层模型:应用层(HTTP/FTP/SMTP/POPS...),传输层(TCP协议),网络层(IP协议,负责为网络上节点分配唯一标识),物理层+数据链路层). IP地址用于标识网络中 ...

  2. android 网络通信方式,Android中基于TCP协议的网络通信

    一.Android网络简介 Android与服务器的通信方式主要有两种,一种是Http通信,另一种是Socket通信. HTTP通信:Android中内置HttpClient,这样可以发方便的发送Ht ...

  3. Android中基于TCP协议的网络通信之使用Socket进行通信

    TCP协议被称为一种端到端的协议.这是因为它为两台计算机的连接起到了重要作用:当一台计算机需要与另一台计算机进行接连时,TCP协议会他们之间建立一个连接:用于发送和接收数据的虚拟链路. TCP协议负责 ...

  4. 基于TCP的网络实时聊天室(socket通信案例)

    开门见山 一.数据结构Map 二.保证线程安全 三.群聊核心方法 四.聊天室具体设计 0.用户登录服务器 1.查看当前上线用户 2.群聊 3.私信 4.退出当前聊天状态 5.离线 6.查看帮助 五.聊 ...

  5. Java进阶:基于TCP的网络实时聊天室(socket通信案例)

    目录 开门见山 一.数据结构Map 二.保证线程安全 三.群聊核心方法 四.聊天室具体设计 0.用户登录服务器 1.查看当前上线用户 2.群聊 3.私信 4.退出当前聊天状态 5.离线 6.查看帮助 ...

  6. QT学习:基于TCP的网络聊天室程序

    TCP与UDP的差别如图: 一.TCP工作原理 如下图所示,TCP能够为应用程序提供可靠的通信连接,使一台计算机发出的字节流无差错 地送达网络上的其他计算机.因此,对可靠性要求高的数据通信系统往往使用 ...

  7. QT示例:基于TCP点对点Socket通讯

    QT示例:基于TCP点对点通讯 一. 概述 二.TCP 协议工作原理 三.TCP 编程模型 四.基于TCP点对点通讯示例 1.客户端 2.客户端Client示例 3.服务器 4.服务器server示例 ...

  8. QT示例:基于TCP 点对多Socket通讯(server,clients)

    QT示例:基于TCP 点对多通讯(server,clients) 一.服务器server 二.客户端Client 下载:基于TCP 点对多Socket通讯 一.服务器server 因为对于客户端来说, ...

  9. socket 网络通信(基于tcp协议)以及粘包解决方案

    socket 网络通信(基于tcp协议)以及粘包解决方案 参考文章: (1)socket 网络通信(基于tcp协议)以及粘包解决方案 (2)https://www.cnblogs.com/amiee- ...

最新文章

  1. Nginx(九)-- Nginx实际使用配置
  2. wxWidgets:wxGenericProgressDialog类用法
  3. 每天一道LeetCode-----找到有多少条连续路径的和为给定值,路径不需要从根节点出发到达叶子节点
  4. 深度剖析 synchronized
  5. 交换二叉树中所有结点的左右子树的位置
  6. Linux解压tar.gz、zip、tar.bz2 文件与对应的命令
  7. __builtin_expect提高运行效率
  8. 【LOJ#10170】国王
  9. 最流行的六大数据模型工具
  10. 电源管理允许此设备唤醒计算机怎么关掉,电脑如何设置电源管理允许鼠标唤醒计算机...
  11. 石溪分校 计算机研究生专业,纽约州立大学石溪分校计算机科学专业
  12. Centos 7.6 Install shc
  13. 使用ML.NET实现健康码识别
  14. TVB十大女星比美十大名花
  15. arcgis中解决点位符号化时重叠冲突
  16. 计算给定字符串中最长回文子串的长度(或查找最长回文子串)
  17. vue 文件.eslintrc.js 配置规则
  18. OpenCV Java入门三 Mat的基本操作
  19. 事件监听函数addEvent
  20. 关联规则与购物篮分析实战

热门文章

  1. 关于物理引擎的使用,及其重力感应的小test
  2. linux通过修改时间排序,linux按修改时间排序
  3. 5.概念(maven,ssm,springMvc,spring,自定义注解,二级缓存,范式,事务,mysql,线程池,map,hashmap,redis,饿汉,懒汉)
  4. 【JavaScript】jQuery 实现耦合的鼠标单击与双击事件
  5. 财报解读:创维集团2022年业绩表现凸显韧性,新能源业务将大有作为
  6. Spark SQL入门:创建SparkSession时import spark.implicits._ 报错: error: value implicits is not a member of...
  7. iPhone快捷指令查询汇率
  8. Tokenview | USDT的印钞和新发行对币价的影响
  9. java netty教程_Netty学习教程之基础使用篇
  10. CSS div从左侧、上侧滑入效果代码实例