[Qt教程] 第37篇 网络(七)TCP(一)

简介:
楼主
  发表于 2013-9-6 15:44:45  | 查看: 398 | 回复: 1
TCP (一)


版权声明

该文章原创于作者yafeilinux,转载请注明出处!


导语


TCP TransmissionControl Protocol ,传输控制协议。与 UDP 不同,它是面向连接和数据流的可靠传输协议。也就是说,它能使一台计算机上的数据无差错的发往网络上的其他计算机,所以当要传输大量数据时,我们选用 TCP 协议。
        TCP协议的程序使用的是客户端/服务器(C/S)模式,在Qt中提供了QTcpSocket类来编写客户端程序,使用QTcpServer类编写服务器端程序。我们在服务器端进行端口的**,一旦发现客户端的连接请求,就会发出newConnection()信号,可以关联这个信号到我们自己的槽进行数据的发送。而在客户端,一旦有数据到来就会发出readyRead()信号,可以关联此信号进行数据的接收。其实,在程序中最难理解的地方就是程序的发送和接收了,为了让大家更好的理解,我们在这一节只是讲述一个传输简单的字符串的例子,在下一节再进行扩展,实现任意文件的传输。



环境: Windows Xp + Qt 4.8.5+Qt Creator2.8.0



目录


一、 服务器端
二、客户端




正文


一、服务器端

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


1.新建QtGui应用
项目名为tcpServer,基类选择QWidget,类名为Widget。完成后打开项目文件tcpServer.pro并添加一行代码:QT += network ,然后保存该文件。


2.widget.ui的设计区添加一个Label,更改其显示文本为“等待连接”,然后更改其objectNamestatusLabel,用于显示一些状态信息。


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))
{    // **本地主机的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.新建Qt Gui应用,
项目名tcpClient,基类选择QWidget,类名为Widget。完成后打开项目文件tcpClient.pro并添加一行代码:QT += network ,然后保存该文件。


2.我们在widget.ui中添加几个标签Label和两个Line Edit以及一个按钮Push Button。设计效果如下图所示。



01.jpg



其中“主机”后的LineEdit objectName hostLineEdit ,“端口号”后的为portLineEdit
“收到的信息”标签的objectName messageLabel 


3.widget.h文件中做更改。
添加头文件: #include   <QtNetwork>
添加private 变量:
QTcpSocket   * tcpSocket;
QString   message;  // 存放从服务器接收到的字符串
quint16blockSize; // 存放文件的大小信息
添加私有槽:
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,点击“连接”按钮,效果如下。

02.jpg

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



结语

        到这里我们最简单的TCP 应用程序就完成了,在下一节我们将会对它进行扩展,实现任意文件的传输。



涉及到的源码:  tcpServer.rar (2.14 KB, 下载次数: 6)  tcpClient.rar (2.54 KB, 下载次数: 6) 



相关文章
|
2天前
|
网络协议 网络架构 数据格式
TCP/IP基础:工作原理、协议栈与网络层
TCP/IP(传输控制协议/互联网协议)是互联网通信的基础协议,支持数据传输和网络连接。本文详细阐述了其工作原理、协议栈构成及网络层功能。TCP/IP采用客户端/服务器模型,通过四个层次——应用层、传输层、网络层和数据链路层,确保数据可靠传输。网络层负责IP寻址、路由选择、分片重组及数据包传输,是TCP/IP的核心部分。理解TCP/IP有助于深入掌握互联网底层机制。
16 2
|
4天前
|
网络协议 Java
谈谈TCP/IP网络编程
【9月更文挑战第1天】在当今数字化的世界中,网络通信是连接各种设备和系统的关键。TCP/IP协议作为互联网通信的基石,被广泛应用于各种网络场景。了解TCP/IP网络编程的概念,并掌握如何在Java中实现TCP/IP通讯,对于开发人员来说是非常重要的。
30 4
|
11天前
|
网络协议 C语言
C语言 网络编程(十三)并发的TCP服务端-以进程完成功能
这段代码实现了一个基于TCP协议的多进程并发服务端和客户端程序。服务端通过创建子进程来处理多个客户端连接,解决了粘包问题,并支持不定长数据传输。客户端则循环发送数据并接收服务端回传的信息,同样处理了粘包问题。程序通过自定义的数据长度前缀确保了数据的完整性和准确性。
|
11天前
|
网络协议 C语言
C语言 网络编程(十一)TCP通信创建流程---服务端
在服务器流程中,新增了绑定IP地址与端口号、建立监听队列及接受连接并创建新文件描述符等步骤。`bind`函数用于绑定IP地址与端口,`listen`函数建立监听队列并设置监听状态,`accept`函数则接受连接请求并创建新的文件描述符用于数据传输。套接字状态包括关闭(CLOSED)、同步发送(SYN-SENT)、同步接收(SYN-RECEIVE)和已建立连接(ESTABLISHED)。示例代码展示了TCP服务端程序如何初始化socket、绑定地址、监听连接请求以及接收和发送数据。
|
11天前
|
网络协议 C语言
C语言 网络编程(十四)并发的TCP服务端-以线程完成功能
这段代码实现了一个基于TCP协议的多线程服务器和客户端程序,服务器端通过为每个客户端创建独立的线程来处理并发请求,解决了粘包问题并支持不定长数据传输。服务器监听在IP地址`172.17.140.183`的`8080`端口上,接收客户端发来的数据,并将接收到的消息添加“-回传”后返回给客户端。客户端则可以循环输入并发送数据,同时接收服务器回传的信息。当输入“exit”时,客户端会结束与服务器的通信并关闭连接。
|
11天前
|
网络协议 C语言
C语言 网络编程(十二)TCP通信创建-粘包
TCP通信中的“粘包”现象指的是由于协议特性,发送方的数据包被拆分并在接收方按序组装,导致多个数据包粘连或单个数据包分割。为避免粘包,可采用定长数据包或先传送数据长度再传送数据的方式。示例代码展示了通过在发送前添加数据长度信息,并在接收时先读取长度后读取数据的具体实现方法。此方案适用于长度不固定的数据传输场景。
|
11天前
|
缓存 网络协议 网络性能优化
C语言 网络编程(二)TCP 协议
TCP(传输控制协议)是一种面向连接、可靠的传输层协议,通过校验和、序列号、确认应答等机制确保数据完整性和可靠性。通信双方需先建立连接,再进行通信,采用三次握手建立连接,四次挥手断开连接。TCP支持任意字节长度的数据传输,具备超时重传、流量控制及拥塞控制机制。三次握手用于同步序列号和确认双方通信能力,四次挥手则确保双方均能完成连接关闭操作,保证数据传输的可靠性。
|
11天前
|
网络协议 C语言
C语言 网络编程(十)TCP通信创建流程---客户端
在TCP通信中,客户端需通过一系列步骤与服务器建立连接并进行数据传输。首先使用 `socket()` 函数创建一个流式套接字,然后通过 `connect()` 函数连接服务器。连接成功后,可以使用 `send()` 和 `recv()` 函数进行数据发送和接收。最后展示了一个完整的客户端示例代码,实现了与服务器的通信过程。
|
17天前
|
C++
C++ Qt开发:QUdpSocket网络通信组件
QUdpSocket是Qt网络编程中一个非常有用的组件,它提供了在UDP协议下进行数据发送和接收的能力。通过简单的方法和信号,可以轻松实现基于UDP的网络通信。不过,需要注意的是,UDP协议本身不保证数据的可靠传输,因此在使用QUdpSocket时,可能需要在应用层实现一些机制来保证数据的完整性和顺序,或者选择在适用的场景下使用UDP协议。
53 2
|
8天前
|
网络协议
网络协议概览:HTTP、UDP、TCP与IP
理解这些基本的网络协议对于任何网络专业人员都是至关重要的,它们不仅是网络通信的基础,也是构建更复杂网络服务和应用的基石。网络技术的不断发展可能会带来新的协议和标准,但这些基本协议的核心概念和原理将继续是理解和创新网络技术的关键。
24 0