TCP即Transmission Control Protocol,传输控制协议。与UDP不同,它是面向连接和数据流的可靠传输协议。也就是说,它能使一台计算机上的数据无差错的发往网络上的其他计算机,所以当要传输大量数据时,我们选用TCP协议。
TCP协议的程序使用的是客户端/服务器模式,在Qt中提供了QTcpSocket类来编写客户端程序,使用QTcpServer类编写服务器端程序。我们在服务器端进行端口的监听,一旦发现客户端的连接请求,就会发出newConnection()信号,我们可以关联这个信号到我们自己的槽函数,进行数据的发送。而在客户端,一旦有数据到来就会发出readyRead()信号,我们可以关联此信号,进行数据的接收。其实,在程序中最难理解的地方就是程序的发送和接收了,为了让大家更好的理解,我们在这一节只是讲述一个传输简单的字符串的例子,在下一节再进行扩展,实现任意文件的传输。
该文档描述了Tcp通信协议的过程。该项目的功能是客户端控制服务器中的灯和蜂鸣器。该协议中设定了灯为0x00,其中灯亮为0x01、灯灭为0x02。设定蜂鸣器为0x01,其中蜂鸣器响为0x01、蜂鸣器关闭为0x02。若成功则返回0x00否则返回0x01。
TCP服务器端:
在服务器端的程序中我们监听任何一台计算机的端口,这里使用的是6666,然后我们关联newConnection()信号和自己写的槽newConnection(),在该槽中我们关联readyRead()信号和自己写的receiveMessage()接收客户端数据的槽。
注:在写通信协议时服务器端要用结构体指针进行编写,这样更方便通信协议的更改。要用结构体指针指向QByteArry 的对象中的数据部分的首地址,可将结构体中的变量与data()中的数据对应,用结构体指针指向结构体变量取出QByteArry中的数据,这就是用结构体来传送数据。
.h中:
1)、定义两个结构体一个是接收客户端发来的数据结构体、此时用结构体指针(CTRL *a;)指向数据首地址,一个是返回客户端结果的结构体、将结果返回给客户端。并相应定义两个私有的结构体指针RESULT *result;CTRL *ctrl;
.h中:两个结构体为预处理命令
#pragma pack(push,1)//为了结构体中变量字节对齐改变了按照以最大字节对
齐规则。接收客户端发来的数据结构体
typedef struct
{
char objects;
char results;
}RESULT;
#pragma pack(pop)
#pragma pack(push,1)//返回客户端结果的结构体
typedef struct
char ctrldevice;
char ctrlorder;
}CTRL;
#pragma pack(pop)
2)、定义QTcpServer对象tcpserver用于监听。定义QTcpSocket*serverreceive 用于接收下一个连接、与指定的端口的计算机建立连接并且往客户端中写入数据。定义的私有成员对象private:
QTcpSocket *serverreceive;
QTcpServer *tcpServer;
CTRL *a;//结构体指针a指向用于接收客户端的数据
QPixmap pixmap;
RESULT *result;
3)、定义公有成员函数他们是控制灯和蜂鸣器,控制灯和蜂鸣器的状态的函数分别调用给客户端发送相应结果的函数newConnect()。
void Ctrl();
void ctrlbulb();灯的亮与灭
void ctrlborn();蜂鸣器的响与不响
4)、定义私有槽
void receiveMessage();接收客户端发送的数据
void newConnect();给客户端发送响应结果
void newconnection();建立下一个连接并实现serverreceive发出信号readyRead()与槽receiveMessage()的连接。
.cpp中
在构造函数中,服务器要监听任何一台计算机的端口号6666,监听时要进行判断。建立新连接newconnection(),接收客户端发送来的数据receiveMessage(),向客户端发送响应结果newConnect()。实现灯和蜂鸣器的控制函数Ctrl()以及实现灯和蜂鸣器开关的函数ctrlbulb()、ctrlborn()。
在构造中
tcpServer = new QTcpServer(this);
if(!tcpServer->listen(QHostAddress::Any,6666))
//服务器监听
//if(!tcpServer->listen(QHostAddress::LocalHost,6666))
{
qDebug() << tcpServer->errorString();
close();
}
//connect(tcpSocket,SIGNAL(readyRead()),SLOT(receiveMessage()));
connect(tcpServer,SIGNAL(newConnection()),this,SLOT(newconnection())) ;
//连接信号和相应槽函数
void Widget::Ctrl()
{
qDebug()<<"111111111";
if(a->ctrldevice==0x00)//灯
{
qDebug()<<"1.11111111111";
result->objects=0x00;//将灯的控制设为0x00
ctrlbulb();
}
if(a->ctrldevice==0x01)
{
qDebug()<<"1.22222222222";
result->objects=0x01;
ctrlborn();
}
}
void Widget::ctrlbulb()//灯
{
qDebug()<<"1.1.1111111111111";
switch(a->ctrlorder)
{
case0x01:
qDebug()<<"1.1.1.1111111111111";
pixmap.load(":/resource/qrc/002.bmp",0,Qt::AvoidDither|Qt::ThresholdA lphaDither|Qt::ThresholdDither);
ui->labelbulb->setPixmap(pixmap);
result->results=0x00;
break;
case0x02:
qDebug()<<"1.1.2.11111111111";
pixmap.load(":/resource/qrc/001.bmp",0,Qt::AvoidDither|Qt::ThresholdA lphaDither|Qt::ThresholdDither);
ui->labelbulb->setPixmap(pixmap);
result->results=0x00;
break;
default:
result->results=0x01;
break;