TCP协议的程序使用的是客户端/服务器模式,在Qt中提供了QTcpSocket类来编写客户端程序,使用QTcpServer类编写服务器端程序。我们在服务器端进行端口的监听,一旦发现客户端的连接请求,就会发出newConnection()信号,我们可以关联这个信号到我们自己的槽函数,进行数据的发送。而在客户端,一旦有数据到来就会发出readyRead()信号,我们可以关联此信号,进行数据的接收。

一、服务器端。

在服务器端的程序中,我们监听本地主机的一个端口,这里使用6666,然后我们关联newConnection()信号与自己写的sendMessage()槽函数。就是说一旦有客户端的连接请求,就会执行sendMessage()函数,在这个函数里我们发送一个简单的字符串。

1.我们新建Qt4 Gui Application,工程名为“tcpServer”,选中QtNetwork模块,Base class选择QWidget。(说明:如果一些Qt Creator版本没有添加模块一项,我们就需要在工程文件tcpServer.pro中添加一行代码:QT += network)

2.我们在widget.ui的设计区添加一个Label,更改其objectName为statusLabel,用于显示一些状态信息。

3.在widget.h文件中做以下更改。

添加头文件:#include <QtNetWork>

添加private对象:QTcpServer *tcpServer;

添加私有槽函数:

private slots:

void sendMessage();

4.在widget.cpp文件中进行更改。

在其构造函数中添加代码:

tcpServer = new QTcpServer(this);

if(!tcpServer->listen(QHostAddress::LocalHost,6666))   //监听其他IP地址则用QHostAddress::Any

{  //监听本地主机的6666端口,如果出错就输出错误信息,并关闭

qDebug() << tcpServer->errorString();

close();

}

connect(tcpServer,SIGNAL(newConnection()),this,SLOT(sendMessage()));

//连接信号和相应槽函数

我们在构造函数中使用tcpServer的listen()函数进行监听,然后关联了newConnection()和我们自己的sendMessage()函数。

下面我们实现sendMessage()函数。

void Widget::sendMessage()

{

QByteArray block; //用于暂存我们要发送的数据

QDataStream out(&block,QIODevice::WriteOnly);

//使用数据流写入数据

out.setVersion(QDataStream::Qt_4_6);

//设置数据流的版本,客户端和服务器端使用的版本要相同

out<<(quint16) 0;

out<<tr(“hello Tcp!!!”);

out.device()->seek(0);

out<<(quint16) (block.size() – sizeof(quint16));

QTcpSocket *clientConnection = tcpServer->nextPendingConnection();

//我们获取已经建立的连接的子套接字

connect(clientConnection,SIGNAL(disconnected()),clientConnection,

SLOT(deleteLater()));

clientConnection->write(block);

clientConnection->disconnectFromHost();

ui->statusLabel->setText(“send message successful!!!”);

//发送数据成功后,显示提示

}

这个是数据发送函数,我们主要介绍两点:

(1)为了保证在客户端能接收到完整的文件,我们都在数据流的最开始写入完整文件的大小信息,这样客户端就可以根据大小信息来判断是否接受到了完整的文件。而在服务器端,我们在发送数据时就要首先发送实际文件的大小信息,但是,文件的大小一开始是无法预知的,所以我们先使用了out<<(quint16) 0;在block的开始添加了一个quint16大小的空间,也就是两字节的空间,它用于后面放置文件的大小信息。然后out<<tr(“hello Tcp!!!”);输入实际的文件,这里是字符串。当文件输入完成后我们在使用out.device()->seek(0);返回到block的开始,加入实际的文件大小信息,也就是后面的代码,它是实际文件的大小:out<<(quint16) (block.size() – sizeof(quint16));

(2)在服务器端我们可以使用tcpServer的nextPendingConnection()函数来获取已经建立的连接的Tcp套接字,使用它来完成数据的发送和其它操作。比如这里,我们关联了disconnected()信号和deleteLater()槽函数,然后我们发送数据

clientConnection->write(block);

然后是clientConnection->disconnectFromHost();它表示当发送完成时就会断开连接,这时就会发出disconnected()信号,而最后调用deleteLater()函数保证在关闭连接后删除该套接字clientConnection。

5.这样服务器的程序就完成了,我们先运行一下程序。

二、客户端。

我们在客户端程序中向服务器发送连接请求,当连接成功时接收服务器发送的数据。

1. .我们新建Qt4 Gui Application,工程名为“tcpClient”,选中QtNetwork模块,Base class选择QWidget。

2,我们在widget.ui中添加几个标签Label和两个Line Edit以及一个按钮Push Button。

其中“主机”后的Line Edit的objectName为hostLineEdit,“端口号”后的为portLineEdit。

“收到的信息”标签的objectName为messageLabel 。

3.在widget.h文件中做更改。

添加头文件:#include <QtNetwork>

添加private变量:

QTcpSocket *tcpSocket;

QString message;  //存放从服务器接收到的字符串

quint16 blockSize;  //存放文件的大小信息

添加私有槽函数:

private slots:

void newConnect(); //连接服务器

void readMessage();  //接收数据

void displayError(QAbstractSocket::SocketError);  //显示错误

4.在widget.cpp文件中做更改。

(1)在构造函数中添加代码:

tcpSocket = new QTcpSocket(this);

connect(tcpSocket,SIGNAL(readyRead()),this,SLOT(readMessage()));

connect(tcpSocket,SIGNAL(error(QAbstractSocket::SocketError)),

this,SLOT(displayError(QAbstractSocket::SocketError)));

这里关联了tcpSocket的两个信号,当有数据到来时发出readyRead()信号,我们执行读取数据的readMessage()函数。当出现错误时发出error()信号,我们执行displayError()槽函数。

(2)实现newConnect()函数。

void Widget::newConnect()

{

blockSize = 0; //初始化其为0

tcpSocket->abort(); //取消已有的连接

tcpSocket->connectToHost(ui->hostLineEdit->text(),

ui->portLineEdit->text().toInt());

//连接到主机,这里从界面获取主机地址和端口号

}

这个函数实现了连接到服务器,下面会在“连接”按钮的单击事件槽函数中调用这个函数。

(3)实现readMessage()函数。

void Widget::readMessage()

{

QDataStream in(tcpSocket);

in.setVersion(QDataStream::Qt_4_6);

//设置数据流版本,这里要和服务器端相同

if(blockSize==0) //如果是刚开始接收数据

{

//判断接收的数据是否有两字节,也就是文件的大小信息

//如果有则保存到blockSize变量中,没有则返回,继续接收数据

if(tcpSocket->bytesAvailable() < (int)sizeof(quint16)) return;

in >> blockSize;

}

if(tcpSocket->bytesAvailable() < blockSize) return;

//如果没有得到全部的数据,则返回,继续接收数据

in >> message;

//将接收到的数据存放到变量中

ui->messageLabel->setText(message);

//显示接收到的数据

}

这个函数实现了数据的接收,它与服务器端的发送函数相对应。首先我们要获取文件的大小信息,然后根据文件的大小来判断是否接收到了完整的文件。

(4)实现displayError()函数。

void Widget::displayError(QAbstractSocket::SocketError)

{

qDebug() << tcpSocket->errorString(); //输出错误信息

}

这里简单的实现了错误信息的输出。

(5)我们在widget.ui中进入“连接”按钮的单击事件槽函数,然后更改如下。

void Widget::on_pushButton_clicked() //连接按钮

{

newConnect(); //请求连接

}

这里直接调用了newConnect()函数。

5.我们运行程序,同时运行服务器程序,然后在“主机”后填入“localhost”,在“端口号”后填入“6666”,点击“连接”按钮。

可以看到我们正确地接收到了数据。因为服务器端和客户端是在同一台机子上运行的,所以我这里填写了“主机”为“localhost”,如果你在不同的机子上运行,需要在“主机”后填写其正确的IP地址。

QT 之 TCP/IP 服务器和客户端(一)相关推荐

  1. linux下用C语言实现TCP/IP服务器与客户端互相发送数据的socket编程

    linux下用C语言实现TCP/IP服务器与客户端互相发送数据的socket编程 server.c #include <sys/stat.h>#include <fcntl.h> ...

  2. python 实战之模仿开发QQ聊天软件(三)TCP/IP服务器与客户端建设

    无论是p2p还是c/s还是b/s,只要用到通讯,必然是要用到今天写的这个. python+MYSQL 在pycharm中的使用 TCP/IP是网络软件最核心的部分,缺少这个你只能当做单机游戏玩. TC ...

  3. Python核心编程(第3版)第2章网络编程中关于tcp/udp服务器和客户端实现代码的运行出错的修正

    在Python核心编程(第3版)第2章网络编程中, 关于tcp/udp服务器和客户端实现代码的运行会出现 ['str' does not support the buffer interface]之类 ...

  4. 西门子SMART200PLC通讯程序集合。 根据工程实践整合200SMART支持的各种通讯,包括和第三方如开放式ISO和TCP的服务器和客户端

    西门子SMART200PLC通讯程序集合. 根据工程实践整合200SMART支持的各种通讯,包括和第三方如开放式ISO和TCP的服务器和客户端,MODBUS通讯等,若工程实践中用到SMART之间或其他 ...

  5. qt之TCP/IP通信客户端与服务器自动连接互相传输数据

    一.前言 在开发上位机软件,用的最多的通信就是网络通信,串口通信和USB通信,串口通信相对是最容易的,USB通信在前面的文章中也专门写了一个博客介绍USB HID通信方式,见链接:https://bl ...

  6. Qt网络程序:基于TCP的服务器、客户端实例

    首先我们需要设置服务器:  项目文件中加入:QT += network  相关头文件: #include<QTcpServer>//监听套接字 #include<QTcpSocket ...

  7. 【python网络编程】创建TCP/UDP服务器进行客户端/服务器间通信

    客户端/服务器网络编程介绍 套接字:通信端点 实例:客户端发送数据,接收服务器返回的时间戳 用Python 编写FTP 客户端程序 客户端/服务器网络编程介绍 软件服务器也运行在一块硬件之上,但是没有 ...

  8. go语言实现tcp并发服务器与客户端

    go语言实现tcp并发服务器端与客户端 server.go // nc 连接, 发送字母, 加收到转大写的字母 // 如果发送exit ,则会断开连接 package mainimport (&quo ...

  9. C++基于TCP/IP简单的客户端、服务器通信程序实例

    本篇文章实现了一个基于TCP 的一个非常简单的客户/服务器通信程序实例.该程序中通讯协议使用的是面向连接的TCP协议SOCK_STREAM, 服务器的ip地址为本地地址即: 127.0.0.1,端口号 ...

最新文章

  1. SuperEdge — Overview
  2. 记事本写python怎么运行-python入门之一个简单记事本
  3. i2s传输中左右声道数据格式
  4. echarts自定义y轴刻度信息
  5. atitit. 日志系统的原则and设计and最佳实践(1)-----原理理论总结.
  6. java面经合集(面试遇到的)
  7. 常见 MIME 类型列表
  8. C4D阿诺德Arnold渲染器介绍
  9. 苹果UDID泄露续:为什么UDID泄露会引起这么大反响?
  10. element表格设置fixed后滚动条滚动时右下角空白块
  11. 漫步者lollipods如何调节音量_Edifer 漫步者 Lolli pods 评测及对一些问题的实际体验...
  12. 文本框、密码框、文本域
  13. python tokenize怎么用_tokenize -- 对 Python 代码使用的标记解析器 — Python 3.7.9 文档...
  14. 学习总结20-04-18:包装类
  15. QT:QSS自定义QLineEdit实例
  16. 读书有益——》祝你今天愉快
  17. pytest之Monkeypatching(猴子补丁)
  18. HEVC(H.265) 基础知识
  19. linux fastq.gz文件如何解压,科学网—批量gzip,ungzip快速压缩和解压fastq文件 - 陈明杰的博文...
  20. 微小区版v11.2.3 小区 物业 智慧

热门文章

  1. Tomcat安装及Eclipse配置教程
  2. HDFS基本概念和特性的详情了解及优缺点
  3. centos7源代码安装mysql5.7_CentOS7源码编译安装MySQL5.7.11
  4. 2021高考襄阳四中成绩查询,襄阳四中2020年高考成绩怎么样(高考喜报)
  5. ccxprocess启动项可以禁用么_Mac怎么禁用Adobe无用自启动项-Mac禁用Adobe无用自启动项方法 - 河东软件园...
  6. java类体_计算机二级考试Java类之类声明以及类体
  7. 理论计算机科学研究生,清华大学理论计算机科学中心姚期智组招收保送研究生,请帮忙发到虎...
  8. python ioc di_Sping(一)——IOC/DI
  9. matlab 2010 工具箱,Matlab2010下使用FULLBNT工具箱實現簡單的靜態貝葉斯網絡及推理...
  10. HDLBits答案(20)_Verilog有限状态机(7)