第十一章 网络编程
Qt网络模块为我们提供了编写TCP / IP客户端和服务器的类。它提供了较低级别的类,例如代表低级网络概念的QTcpSocket,QTcpServer和QUdpSocket,以及诸如QNetworkRequest,QNetworkReply和QNetworkAccessManager之类的高级类来执行使用通用协议的网络操作。 它还提供了诸如QNetworkConfiguration,QNetworkConfigurationManager和QNetworkSession等类,实现承载管理。 想要在程序中使用Qt网络模块,我们需要在pro项目配置文件里增加下面的一条语句。 QT += network
11.1 获取本机的网络信息 为什么先写获取本机网络信息的内容呢?在建立网络通信之前我们至少得获取对方的IP地址。在网络应用中,经常需要用到本机的主机名、IP地址、MAC地址等网络信息,通常通在Windows通过调出命令行cmd窗口输入ipconfig或者在Linux系统中使用ifconfig命令就可以查看相关信息了,在这里我们利用Qt做出一个可以查询的界面和功能出来,为了后面的网络编程打下一个简单的基础。 Qt提供了QHostInfo和QNetworkInterface类可以用于此类信息查询。更多关于QHostInfo和QNetworkInterface的相关函数可以在Qt的帮助文档中找到。下面我们写代码时会使用到相关的函数,有清楚的注释。 11.1.1 应用实例 本例目的:了解如何通过QHostInfo和QNetworkInterface类获取本地网络所有接口的信息。 例07_networkhostinfo,获取本机网络接口信息(难度:一般)。项目路径为Qt/2/07_networkhostinfo。本例获取本机的网络接口信息,打印在文本浏览框上,点击按钮可直接获取,为了清楚看见是重新获取的过程,本例点击获取本机信息按钮后延时1s去刷新获取的信息。点击另一个清空文本信息按钮可以清空文本浏览框上的文本内容。 项目文件07_networkhostinfo.pro文件第一行添加的代码部分如下。 07_ networkhostinfo.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QPushButton>
6 #include <QTextBrowser>
7 #include <QVBoxLayout>
8 #include <QHBoxLayout>
9 #include <QTimer>
10
11 class MainWindow : public QMainWindow
12 {
13 Q_OBJECT
14
15 public:
16 MainWindow(QWidget *parent = nullptr);
17 ~MainWindow();
18
19 private:
20
21 QPushButton *pushButton[2];
22
23
24 QTextBrowser *textBrowser;
25
26
27 QWidget *hWidget;
28 QWidget *vWidget;
29
30
31 QHBoxLayout *hBoxLayout;
32 QVBoxLayout *vBoxLayout;
33
34
35 QTimer *timer;
36
37
38 QString getHostInfo();
39
40 private slots:
41
42 void timerTimeOut();
43
44
45 void showHostInfo();
46
47
48 void timerStart();
49
50
51 void clearHostInfo();
52 };
53 #endif
54
头文件里主要是声明两个按钮和一个文本浏览框。另外还有一个定时器,声明一些槽函数,比较简单。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2 #include <QNetworkInterface>
3 #include <QHostInfo>
4 #include <QThread>
5 #include <QDebug>
6
7 MainWindow::MainWindow(QWidget *parent)
8 : QMainWindow(parent)
9 {
10
11 this->setGeometry(0, 0, 800, 480);
12
13
14 pushButton[0] = new QPushButton();
15 pushButton[1] = new QPushButton();
16
17 pushButton[0]->setText("获取本机信息");
18 pushButton[1]->setText("清空文本信息");
19
20
22 pushButton[0]->setSizePolicy(QSizePolicy::Fixed,
23 QSizePolicy::Fixed);
24 pushButton[1]->setSizePolicy(QSizePolicy::Fixed,
25 QSizePolicy::Fixed);
26
27
28 hWidget = new QWidget();
29 vWidget = new QWidget();
30
31
32 hBoxLayout = new QHBoxLayout();
33 vBoxLayout = new QVBoxLayout();
34
35
36 textBrowser = new QTextBrowser();
37
38
39 hBoxLayout->addWidget(pushButton[0]);
40 hBoxLayout->addWidget(pushButton[1]);
41
42
43 hWidget->setLayout(hBoxLayout);
44
45
46 vBoxLayout->addWidget(textBrowser);
47 vBoxLayout->addWidget(hWidget);
48
49
50 vWidget->setLayout(vBoxLayout);
51
52
53 setCentralWidget(vWidget);
54
55
56 timer = new QTimer();
57
58
59 connect(pushButton[0], SIGNAL(clicked()),
60 this, SLOT(timerStart()));
61 connect(pushButton[1], SIGNAL(clicked()),
62 this, SLOT(clearHostInfo()));
63 connect(timer, SIGNAL(timeout()),
64 this, SLOT(timerTimeOut()));
65 }
66
67 MainWindow::~MainWindow()
68 {
69 }
70
71
72 void MainWindow::timerStart()
73 {
74
75 textBrowser->clear();
76
77
78 timer->start(1000);
79 }
80
81 void MainWindow::timerTimeOut()
82 {
83
84 showHostInfo();
85
86
87 timer->stop();
88 }
89
90 QString MainWindow::getHostInfo()
91 {
92
93 QString str = "主机名称:" + QHostInfo::localHostName() + "\n";
94
95
97 QList<QNetworkInterface> list
98 = QNetworkInterface::allInterfaces();
99
100
101 foreach (QNetworkInterface interface, list) {
102 str+= "网卡设备:" + interface.name() + "\n";
103 str+= "MAC地址:" + interface.hardwareAddress() + "\n";
104
105
106 QList<QNetworkAddressEntry> entryList
107 = interface.addressEntries();
108
109
110 foreach (QNetworkAddressEntry entry, entryList) {
111
112 if (entry.ip().protocol() ==
113 QAbstractSocket::IPv4Protocol) {
114 str+= "IP 地址:" + entry.ip().toString() + "\n";
115 str+= "子网掩码:" + entry.netmask().toString() + "\n";
116 str+= "广播地址:" + entry.broadcast().toString() + "\n\n";
117 }
118 }
119 }
120
121
122 return str;
123 }
124
125 void MainWindow::showHostInfo()
126 {
127
128 textBrowser->insertPlainText(getHostInfo());
129 }
130
131 void MainWindow::clearHostInfo()
132 {
133
134 if (!textBrowser->toPlainText().isEmpty())
135
136
137 textBrowser->clear();
138 }
第90~123行,是本例最重要的代码。
第93行,通过QHostInfo的localHostName函数获取主机名称。
第97~98行,通过QNetworkInterface::allInterfaces()获取网络接口列表list类存储IP地址子网掩码和广播地址。如果我们用qDebug()函数打印出list,可以发现获取了所有的网络信息。而我们要提取网络里面的网络信息使用QNetworkAddressEntry。
第106~107行,使用QNetworkAddressEntry从interface接口里使用函数addressEntries(),获取所有的条目。就可以使用QNetworkAddressEntry的对象entry获取IP地址子网掩码和广播地址。
第110~118行,因为获取的entries在一个QNetworkInterface下可能有两个IP,分别是ipv4和ipv6。这里使用ip().protocol()来判断协议的类型,只留下ipv4类型的信息。筛选信息在我们写程序常常需要的。 11.1.2 程序运行效果 点击获取本机信息,在文本浏览框内就打印出本机的网络信息(包括了主机名,网卡名,ip地址等)。这里因为过滤掉了IPv6的信息。通常一个网卡有两个ip地址,一个是ipv4,另一个是ipv6的地址。下面的网卡设备lo,是本地回环网卡。另一个ens33是虚拟机的网卡,由VMware虚拟出来的。点击清空文本信息会清空文本浏览框里的网络信息。
11.2 TCP通信 11.2.1 TCP简介 TCP协议(Transmission Control Protocol)全称是传输控制协议是一种面向连接的、可靠的、基于字节流的传输层通信协议。 TCP通信必须先建立TCP连接,通信端分为客户端和服务端。服务端通过监听某个端口来监听是否有客户端连接到来,如果有连接到来,则建立新的socket连接;客户端通过ip和port连接服务端,当成功建立连接之后,就可进行数据的收发了。需要注意的是,在Qt中,Qt把socket当成输入输出流来对待的,数据的收发是通过read()和write()来进行的,需要与我们常见的send()与recv()进行区分。 TCP客户端与服务端通信示意图如下。
11.2.2 TCP服务端应用实例 本例目的:了解TCP服务端的使用。 例08_tcpserver,TCP服务端(难度:一般)。项目路径为Qt/2/08_tcpserver。本例大体流程首先获取本地IP地址。创建一个tcpSocket套接字,一个tcpServer服务端。点击监听即监听本地的主机IP地址和端口,同时等待服务端的连接。此程序需要结合客户端一起使用。 项目文件08_tcpserver.pro文件第一行添加的代码部分如下。 08_tcpserver.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QTcpServer>
6 #include <QTcpSocket>
7 #include <QVBoxLayout>
8 #include <QHBoxLayout>
9 #include <QPushButton>
10 #include <QTextBrowser>
11 #include <QLabel>
12 #include <QComboBox>
13 #include <QSpinBox>
14 #include <QHostInfo>
15 #include <QLineEdit>
16 #include <QNetworkInterface>
17 #include <QDebug>
18
19 class MainWindow : public QMainWindow
20 {
21 Q_OBJECT
22
23 public:
24 MainWindow(QWidget *parent = nullptr);
25 ~MainWindow();
26
27 private:
28
29 QTcpServer *tcpServer;
30
31
32 QTcpSocket *tcpSocket;
33
34
35 QPushButton *pushButton[4];
36
37
38 QLabel *label[2];
39
40
41 QWidget *hWidget[3];
42
43
44 QHBoxLayout *hBoxLayout[3];
45
46
47 QWidget *vWidget;
48
49
50 QVBoxLayout *vBoxLayout;
51
52
53 QTextBrowser *textBrowser;
54
55
56 QComboBox *comboBox;
57
58
59 QSpinBox *spinBox;
60
61
62 QLineEdit *lineEdit;
63
64
65 QList<QHostAddress> IPlist;
66
67
68 void getLocalHostIP();
69
70 private slots:
71
72 void clientConnected();
73
74
75 void startListen();
76
77
78 void stopListen();
79
80
81 void clearTextBrowser();
82
83
84 void receiveMessages();
85
86
87 void sendMessages();
88
89
90 void socketStateChange(QAbstractSocket::SocketState);
91 };
92 #endif
头文件里主要是声明界面用的元素,及一些槽函数。重点是声明tcpServer和tcpSocket。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2
3 MainWindow::MainWindow(QWidget *parent)
4 : QMainWindow(parent)
5 {
6
7 this->setGeometry(0, 0, 800, 480);
8
9
10 tcpServer = new QTcpServer(this);
11 tcpSocket = new QTcpSocket(this);
12
13
14 pushButton[0] = new QPushButton();
15
16 pushButton[1] = new QPushButton();
17
18 pushButton[2] = new QPushButton();
19
20 pushButton[3] = new QPushButton();
21
22
23 hBoxLayout[0] = new QHBoxLayout();
24
25 hBoxLayout[1] = new QHBoxLayout();
26
27 hBoxLayout[2] = new QHBoxLayout();
28
29 hBoxLayout[3] = new QHBoxLayout();
30
31
32 hWidget[0] = new QWidget();
33
34 hWidget[1] = new QWidget();
35
36 hWidget[2] = new QWidget();
37
38 vWidget = new QWidget();
39 vBoxLayout = new QVBoxLayout();
40
41
42 label[0] = new QLabel();
43 label[1] = new QLabel();
44
45 lineEdit = new QLineEdit();
46 comboBox = new QComboBox();
47 spinBox = new QSpinBox();
48 textBrowser = new QTextBrowser();
49
50 label[0]->setText("监听IP地址:");
51 label[1]->setText("监听端口:");
52
53
54 label[0]->setSizePolicy(QSizePolicy::Fixed,
55 QSizePolicy::Fixed);
56 label[1]->setSizePolicy(QSizePolicy::Fixed,
57 QSizePolicy::Fixed);
58
59
60 spinBox->setRange(10000, 99999);
61
62 pushButton[0]->setText("开始监听");
63 pushButton[1]->setText("停止监听");
64 pushButton[2]->setText("清空文本");
65 pushButton[3]->setText("发送消息");
66
67
68 pushButton[1]->setEnabled(false);
69
70
71 lineEdit->setText("www.openedv.com正点原子论坛");
72
73
74 hBoxLayout[0]->addWidget(pushButton[0]);
75 hBoxLayout[0]->addWidget(pushButton[1]);
76 hBoxLayout[0]->addWidget(pushButton[2]);
77
78
79 hWidget[0]->setLayout(hBoxLayout[0]);
80
81
82 hBoxLayout[1]->addWidget(label[0]);
83 hBoxLayout[1]->addWidget(comboBox);
84 hBoxLayout[1]->addWidget(label[1]);
85 hBoxLayout[1]->addWidget(spinBox);
86
87
88 hWidget[1]->setLayout(hBoxLayout[1]);
89
90
91 hBoxLayout[2]->addWidget(lineEdit);
92 hBoxLayout[2]->addWidget(pushButton[3]);
93
94
95 hWidget[2]->setLayout(hBoxLayout[2]);
96
97
98 vBoxLayout->addWidget(textBrowser);
99 vBoxLayout->addWidget(hWidget[1]);
100 vBoxLayout->addWidget(hWidget[0]);
101 vBoxLayout->addWidget(hWidget[2]);
102
103
104 vWidget->setLayout(vBoxLayout);
105
106
107 setCentralWidget(vWidget);
108
109
110 getLocalHostIP();
111
112
113 connect(pushButton[0], SIGNAL(clicked()),
114 this, SLOT(startListen()));
115 connect(pushButton[1], SIGNAL(clicked()),
116 this, SLOT(stopListen()));
117 connect(pushButton[2], SIGNAL(clicked()),
118 this, SLOT(clearTextBrowser()));
119 connect(pushButton[3], SIGNAL(clicked()),
120 this, SLOT(sendMessages()));
121 connect(tcpServer, SIGNAL(newConnection()),
122 this, SLOT(clientConnected()));
123 }
124
125 MainWindow::~MainWindow()
126 {
127 }
128
129
130 void MainWindow::clientConnected()
131 {
132
133 tcpSocket = tcpServer->nextPendingConnection();
134
135 QString ip = tcpSocket->peerAddress().toString();
136
137 quint16 port = tcpSocket->peerPort();
138
139 textBrowser->append("客户端已连接");
140 textBrowser->append("客户端ip地址:"
141 + ip);
142 textBrowser->append("客户端端口:"
143 + QString::number(port));
144
145 connect(tcpSocket, SIGNAL(readyRead()),
146 this, SLOT(receiveMessages()));
147 connect(tcpSocket,
148 SIGNAL(stateChanged(QAbstractSocket::SocketState)),
149 this,
150 SLOT(socketStateChange(QAbstractSocket::SocketState)));
151 }
152
153
154 void MainWindow::getLocalHostIP()
155 {
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
175 QList<QNetworkInterface> list
176 = QNetworkInterface::allInterfaces();
177
178
179 foreach (QNetworkInterface interface, list) {
180
181
182 QList<QNetworkAddressEntry> entryList
183 = interface.addressEntries();
184
185
186 foreach (QNetworkAddressEntry entry, entryList) {
187
188 if (entry.ip().protocol() ==
189 QAbstractSocket::IPv4Protocol) {
190 comboBox->addItem(entry.ip().toString());
191
192 IPlist<<entry.ip();
193 }
194 }
195 }
196 }
197
198
199 void MainWindow::startListen()
200 {
201
202 if (comboBox->currentIndex() != -1) {
203 qDebug()<<"start listen"<<endl;
204 tcpServer->listen(IPlist[comboBox->currentIndex()],
205 spinBox->value());
206
207
208 pushButton[0]->setEnabled(false);
209 pushButton[1]->setEnabled(true);
210 comboBox->setEnabled(false);
211 spinBox->setEnabled(false);
212
213
214 textBrowser->append("服务器IP地址:"
215 + comboBox->currentText());
216 textBrowser->append("正在监听端口:"
217 + spinBox->text());
218 }
219 }
220
221
222 void MainWindow::stopListen()
223 {
224 qDebug()<<"stop listen"<<endl;
225
226 tcpServer->close();
227
228
230 if (tcpSocket->state() == tcpSocket->ConnectedState)
231 tcpSocket->disconnectFromHost();
232
233
234 pushButton[1]->setEnabled(false);
235 pushButton[0]->setEnabled(true);
236 comboBox->setEnabled(true);
237 spinBox->setEnabled(true);
238
239
240 textBrowser->append("已停止监听端口:"
241 + spinBox->text());
242 }
243
244
245 void MainWindow::clearTextBrowser()
246 {
247
248 textBrowser->clear();
249 }
250
251
252 void MainWindow::receiveMessages()
253 {
254
255 QString messages = "客户端:" + tcpSocket->readAll();
256 textBrowser->append(messages);
257 }
258
259
260 void MainWindow::sendMessages()
261 {
262 if(NULL == tcpSocket)
263 return;
264
265
266 if(tcpSocket->state() == tcpSocket->ConnectedState) {
267
268 tcpSocket->write(lineEdit->text().toUtf8().data());
269
270
271 textBrowser->append("服务端:" + lineEdit->text());
272 }
273 }
274
275
276 void MainWindow::socketStateChange(QAbstractSocket::SocketState state)
277 {
278 switch (state) {
279 case QAbstractSocket::UnconnectedState:
280 textBrowser->append("scoket状态:UnconnectedState");
281 break;
282 case QAbstractSocket::ConnectedState:
283 textBrowser->append("scoket状态:ConnectedState");
284 break;
285 case QAbstractSocket::ConnectingState:
286 textBrowser->append("scoket状态:ConnectingState");
287 break;
288 case QAbstractSocket::HostLookupState:
289 textBrowser->append("scoket状态:HostLookupState");
290 break;
291 case QAbstractSocket::ClosingState:
292 textBrowser->append("scoket状态:ClosingState");
293 break;
294 case QAbstractSocket::ListeningState:
295 textBrowser->append("scoket状态:ListeningState");
296 break;
297 case QAbstractSocket::BoundState:
298 textBrowser->append("scoket状态:BoundState");
299 break;
300 default:
301 break;
302 }
303 }
上面的代码主要是服务端开启监听,如果有客户端连到服务端,就会发射newConnection()信号,同时也连接到接收消息的信号与槽函数。点击发送消息按钮就可以使用tcpSocket发送消息。注意发送消息和接收消息都是通过tcpSocket的read()和write()进行。 11.2.3 TCP客户端应用实例 本例目的:了解TCP客户的使用。 例09_tcpclient,TCP客户端(难度:一般)。项目路径为Qt/2/09_ tcpclient。本例大体流程:首先获取本地IP地址。创建一个tcpSocket套接字,然后用tcpSocket套接字使用connectToHost函数连接服务端的主机IP地址和端口,即可相互通信。 项目文件08_tcpserver.pro文件第一行添加的代码部分如下。 09_tcpclient.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QTcpServer>
6 #include <QTcpSocket>
7 #include <QVBoxLayout>
8 #include <QHBoxLayout>
9 #include <QPushButton>
10 #include <QTextBrowser>
11 #include <QLabel>
12 #include <QComboBox>
13 #include <QSpinBox>
14 #include <QHostInfo>
15 #include <QLineEdit>
16 #include <QNetworkInterface>
17 #include <QDebug>
18
19 class MainWindow : public QMainWindow
20 {
21 Q_OBJECT
22
23 public:
24 MainWindow(QWidget *parent = nullptr);
25 ~MainWindow();
26
27 private:
28
29 QTcpSocket *tcpSocket;
30
31
32 QPushButton *pushButton[4];
33
34
35 QLabel *label[2];
36
37
38 QWidget *hWidget[3];
39
40
41 QHBoxLayout *hBoxLayout[3];
42
43
44 QWidget *vWidget;
45
46
47 QVBoxLayout *vBoxLayout;
48
49
50 QTextBrowser *textBrowser;
51
52
53 QComboBox *comboBox;
54
55
56 QSpinBox *spinBox;
57
58
59 QLineEdit *lineEdit;
60
61
62 QList<QHostAddress> IPlist;
63
64
65 void getLocalHostIP();
66
67 private slots:
68
69 void toConnect();
70
71
72 void toDisConnect();
73
74
75 void connected();
76
77
78 void disconnected();
79
80
81 void clearTextBrowser();
82
83
84 void receiveMessages();
85
86
87 void sendMessages();
88
89
90 void socketStateChange(QAbstractSocket::SocketState);
91 };
92 #endif
头文件里主要是声明界面用的元素,及一些槽函数。重点是声明tcpSocket。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2
3 MainWindow::MainWindow(QWidget *parent)
4 : QMainWindow(parent)
5 {
6
7 this->setGeometry(0, 0, 800, 480);
8
9
10 tcpSocket = new QTcpSocket(this);
11
12
13 pushButton[0] = new QPushButton();
14
15 pushButton[1] = new QPushButton();
16
17 pushButton[2] = new QPushButton();
18
19 pushButton[3] = new QPushButton();
20
21
22 hBoxLayout[0] = new QHBoxLayout();
23
24 hBoxLayout[1] = new QHBoxLayout();
25
26 hBoxLayout[2] = new QHBoxLayout();
27
28 hBoxLayout[3] = new QHBoxLayout();
29
30
31 hWidget[0] = new QWidget();
32
33 hWidget[1] = new QWidget();
34
35 hWidget[2] = new QWidget();
36
37
38 vWidget = new QWidget();
39 vBoxLayout = new QVBoxLayout();
40
41
42 label[0] = new QLabel();
43 label[1] = new QLabel();
44
45 lineEdit = new QLineEdit();
46 comboBox = new QComboBox();
47 spinBox = new QSpinBox();
48 textBrowser = new QTextBrowser();
49
50 label[0]->setText("服务器地址:");
51 label[1]->setText("服务器端口:");
52
53
54 label[0]->setSizePolicy(QSizePolicy::Fixed,
55 QSizePolicy::Fixed);
56 label[1]->setSizePolicy(QSizePolicy::Fixed,
57 QSizePolicy::Fixed);
58
59
60 spinBox->setRange(10000, 99999);
61
62 pushButton[0]->setText("连接服务器");
63 pushButton[1]->setText("断开连接");
64 pushButton[2]->setText("清空文本");
65 pushButton[3]->setText("发送消息");
66
67
68 pushButton[1]->setEnabled(false);
69
70
71 lineEdit->setText("广州星翼电子科技有限公司");
72
73
74 hBoxLayout[0]->addWidget(pushButton[0]);
75 hBoxLayout[0]->addWidget(pushButton[1]);
76 hBoxLayout[0]->addWidget(pushButton[2]);
77
78
79 hWidget[0]->setLayout(hBoxLayout[0]);
80
81 hBoxLayout[1]->addWidget(label[0]);
82 hBoxLayout[1]->addWidget(comboBox);
83 hBoxLayout[1]->addWidget(label[1]);
84 hBoxLayout[1]->addWidget(spinBox);
85
86
87 hWidget[1]->setLayout(hBoxLayout[1]);
88
89
90 hBoxLayout[2]->addWidget(lineEdit);
91 hBoxLayout[2]->addWidget(pushButton[3]);
92
93
94 hWidget[2]->setLayout(hBoxLayout[2]);
95
96
97 vBoxLayout->addWidget(textBrowser);
98 vBoxLayout->addWidget(hWidget[1]);
99 vBoxLayout->addWidget(hWidget[0]);
100 vBoxLayout->addWidget(hWidget[2]);
101
102
103 vWidget->setLayout(vBoxLayout);
104
105
106 setCentralWidget(vWidget);
107
108
109 getLocalHostIP();
110
111
112 connect(pushButton[0], SIGNAL(clicked()),
113 this, SLOT(toConnect()));
114 connect(pushButton[1], SIGNAL(clicked()),
115 this, SLOT(toDisConnect()));
116 connect(pushButton[2], SIGNAL(clicked()),
117 this, SLOT(clearTextBrowser()));
118 connect(pushButton[3], SIGNAL(clicked()),
119 this, SLOT(sendMessages()));
120 connect(tcpSocket, SIGNAL(connected()),
121 this, SLOT(connected()));
122 connect(tcpSocket, SIGNAL(disconnected()),
123 this, SLOT(disconnected()));
124 connect(tcpSocket, SIGNAL(readyRead()),
125 this, SLOT(receiveMessages()));
126 connect(tcpSocket,
127 SIGNAL(stateChanged(QAbstractSocket::SocketState)),
128 this,
129 SLOT(socketStateChange(QAbstractSocket::SocketState)));
130 }
131
132 MainWindow::~MainWindow()
133 {
134 }
135
136 void MainWindow::toConnect()
137 {
138
139 if (tcpSocket->state() != tcpSocket->ConnectedState) {
140
141 tcpSocket->connectToHost(IPlist[comboBox->currentIndex()],
142 spinBox->value());
143 }
144 }
145
146 void MainWindow::toDisConnect()
147 {
148
149 tcpSocket->disconnectFromHost();
150
151
152 tcpSocket->close();
153 }
154
155 void MainWindow::connected()
156 {
157
158 textBrowser->append("已经连上服务端");
159
160
161 pushButton[0]->setEnabled(false);
162 pushButton[1]->setEnabled(true);
163 comboBox->setEnabled(false);
164 spinBox->setEnabled(false);
165 }
166
167 void MainWindow::disconnected()
168 {
169
170 textBrowser->append("已经断开服务端");
171
172
173 pushButton[1]->setEnabled(false);
174 pushButton[0]->setEnabled(true);
175 comboBox->setEnabled(true);
176 spinBox->setEnabled(true);
177 }
178
179
180 void MainWindow::getLocalHostIP()
181 {
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
201 QList<QNetworkInterface> list
202 = QNetworkInterface::allInterfaces();
203
204
205 foreach (QNetworkInterface interface, list) {
206
207
208 QList<QNetworkAddressEntry> entryList
209 = interface.addressEntries();
210
211
212 foreach (QNetworkAddressEntry entry, entryList) {
213
214 if (entry.ip().protocol() ==
215 QAbstractSocket::IPv4Protocol) {
216 comboBox->addItem(entry.ip().toString());
217
218 IPlist<<entry.ip();
219 }
220 }
221 }
222 }
223
224
225 void MainWindow::clearTextBrowser()
226 {
227
228 textBrowser->clear();
229 }
230
231
232 void MainWindow::receiveMessages()
233 {
234
235 QString messages = tcpSocket->readAll();
236 textBrowser->append("服务端:" + messages);
237 }
238
239
240 void MainWindow::sendMessages()
241 {
242 if(NULL == tcpSocket)
243 return;
244
245 if(tcpSocket->state() == tcpSocket->ConnectedState) {
246
247 textBrowser->append("客户端:" + lineEdit->text());
248
249
250 tcpSocket->write(lineEdit->text().toUtf8().data());
251 }
252 }
253
254
255 void MainWindow::socketStateChange(QAbstractSocket::SocketState state)
256 {
257 switch (state) {
258 case QAbstractSocket::UnconnectedState:
259 textBrowser->append("scoket状态:UnconnectedState");
260 break;
261 case QAbstractSocket::ConnectedState:
262 textBrowser->append("scoket状态:ConnectedState");
263 break;
264 case QAbstractSocket::ConnectingState:
265 textBrowser->append("scoket状态:ConnectingState");
266 break;
267 case QAbstractSocket::HostLookupState:
268 textBrowser->append("scoket状态:HostLookupState");
269 break;
270 case QAbstractSocket::ClosingState:
271 textBrowser->append("scoket状态:ClosingState");
272 break;
273 case QAbstractSocket::ListeningState:
274 textBrowser->append("scoket状态:ListeningState");
275 break;
276 case QAbstractSocket::BoundState:
277 textBrowser->append("scoket状态:BoundState");
278 break;
279 default:
280 break;
281 }
282 }
上面的代码主要是客户端开使用connectToHost通过IP地址和端口与服务端连接,如果连接成功,就会发射connected ()信号,同时也连接到接收消息的信号与槽函数。点击发送消息按钮就可以使用tcpSocket发送消息。注意发送消息和接收消息都是通过tcpSocket的read()和write()进行。 11.2.4 程序运行效果 开启服务端后,需要选择本地监听的IP地址和监听的端口(特别需要注意,不要选择监听的端口与本地主机的已经使用的端口,所以编者把端口号设置的特别大,查看本地已经使用的端口号可以使用netstat指令。) 启动客户端后,选择需要连接的服务器IP地址和服务器监听的端口。点击连接后就可以相互发送消息了。 注意服务端和客户端都本例都是选择了本地环回IP 127.0.0.1测试。也可以选择本地的其他IP地址进行测试。 TCP服务端:
TCP客户端:
11.3 UDP通信 11.3.1 UDP简介 UDP(User Datagram Protocol即用户数据报协议)是一个轻量级的,不可靠的,面向数据报的无连接协议。我们日常生活中使用的QQ,其聊天时的文字内容是使用UDP协议进行消息发送的。因为QQ有很多用户,发送的大部分都是短消息,要求能及时响应,并且对安全性要求不是很高的情况下使用UDP协议。但是QQ也并不是完全使用UDP协议,比如我们在传输文件时就会选择TCP协议,保证文件正确传输。像QQ语音和QQ视频通话,UDP的优势就很突出了。在选择使用协议的时候,选择UDP必须要谨慎。在网络质量令人十分不满意的环境下,UDP协议数据包丢失会比较严重。但是由于UDP的特性:它不属于连接型协议,因而具有资源消耗小,处理速度快的优点,所以通常音频、视频和普通数据在传送时使用UDP较多,因为它们即使偶尔丢失一两个数据包,也不会对接收结果产生太大影响。 QUdpSocket类提供了一个UDP套接字。QUdpSocket是QAbstractSocket的子类,允许发送和接收UDP数据报。使用该类最常见的方法是使用bind()绑定到一个地址和端口,然后调用writeDatagram()和readDatagram() / receiveDatagram()来传输数据。注意发送数据一般少于512字节。如果发送多于512字节的数据,即使我们发送成功了,也会在IP层被分片(分成小片段)。 如果您想使用标准的QIODevice函数read()、readLine()、write()等,您必须首先通过调用connectToHost()将套接字直接连接到对等体。每次将数据报写入网络时,套接字都会发出bytesWritten()信号。 如果您只是想发送数据报,您不需要调用bind()。readyRead()信号在数据报到达时发出。在这种情况下,hasPendingDatagrams()返回true。调用pendingDatagramSize()来获取第一个待处理数据报的大小,并调用readDatagram()或receiveDatagram()来读取它。注意:当您接收到readyRead()信号时,一个传入的数据报应该被读取,否则这个信号将不会被发送到下一个数据报。 UDP通信示意图如下。重点是QUdpSocket类,已经为我们提供了UDP通信的基础。
UDP消息传送有三种模式,分别是单播、广播和组播三种模式。
?单播(unicast):单播用于两个主机之间的端对端通信,需要知道对方的IP地址与端口。 ?广播(broadcast):广播UDP与单播UDP的区别就是IP地址不同,广播一般使用广播地址255.255.255.255,将消息发送到在同一广播(也就是局域网内同一网段)网络上的每个主机。值得强调的是:本地广播信息是不会被路由器转发。当然这是十分容易理解的,因为如果路由器转发了广播信息,那么势必会引起网络瘫痪。这也是为什么IP协议的设计者故意没有定义互联网范围的广播机制。广播地址通常用于在网络游戏中处于同一本地网络的玩家之间交流状态信息等。其实广播顾名思义,就是想局域网内所有的人说话,但是广播还是要指明接收者的端口号的,因为不可能接受者的所有端口都来收听广播。 ?组播(multicast):组播(多点广播),也称为“多播”,将网络中同一业务类型主机进行了逻辑上的分组,进行数据收发的时候其数据仅仅在同一分组中进行,其他的主机没有加入此分组不能收发对应的数据。在广域网上广播的时候,其中的交换机和路由器只向需要获取数据的主机复制并转发数据。主机可以向路由器请求加入或退出某个组,网络中的路由器和交换机有选择地复制并传输数据,将数据仅仅传输给组内的主机。多播的这种功能,可以一次将数据发送到多个主机,又能保证不影响其他不需要(未加入组)的主机的其他通信。 注意:单播一样和多播是允许在广域网即Internet上进行传输的,而广播仅仅在同一局域网上才能进行。 11.3.2 UDP单播与广播 广播UDP与单播UDP的区别就是IP地址不同,所以我们的实例可以写成一个。我们可以这么理解,单播实际上是通信上对应一对一,广播则是一对多(多,这里指广播地址内的所有主机)。 11.3.2.1 应用实例 本例目的:了解QUdpSocket单播和广播使用。 例10_udp_unicast_broadcast,UDP单播与广播应用(难度:一般)。项目路径为Qt/2/10_udp_unicast_broadcast。本例大体流程首先获取本地IP地址。创建一个udpSocket套接字,然后绑定本地主机的端口(也就是监听端口)。我们可以使用QUdpSocket类提供的读写函数readDatagram和writeDatagram,知道目标IP地址和端口,即可完成消息的接收与发送。 项目文件10_udp_unicast_broadcast.pro文件第一行添加的代码部分如下。 10_udp_unicast_broadcast.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QUdpSocket>
6 #include <QVBoxLayout>
7 #include <QHBoxLayout>
8 #include <QPushButton>
9 #include <QTextBrowser>
10 #include <QLabel>
11 #include <QComboBox>
12 #include <QSpinBox>
13 #include <QHostInfo>
14 #include <QLineEdit>
15 #include <QNetworkInterface>
16 #include <QDebug>
17
18 class MainWindow : public QMainWindow
19 {
20 Q_OBJECT
21
22 public:
23 MainWindow(QWidget *parent = nullptr);
24 ~MainWindow();
25
26 private:
27
28 QUdpSocket *udpSocket;
29
30
31 QPushButton *pushButton[5];
32
33
34 QLabel *label[3];
35
36
37 QWidget *hWidget[3];
38
39
40 QHBoxLayout *hBoxLayout[3];
41
42
43 QWidget *vWidget;
44
45
46 QVBoxLayout *vBoxLayout;
47
48
49 QTextBrowser *textBrowser;
50
51
52 QComboBox *comboBox;
53
54
55 QSpinBox *spinBox[2];
56
57
58 QLineEdit *lineEdit;
59
60
61 QList<QHostAddress> IPlist;
62
63
64 void getLocalHostIP();
65
66 private slots:
67
68 void bindPort();
69
70
71 void unbindPort();
72
73
74 void clearTextBrowser();
75
76
77 void receiveMessages();
78
79
80 void sendMessages();
81
82
83 void sendBroadcastMessages();
84
85
86 void socketStateChange(QAbstractSocket::SocketState);
87 };
88 #endif
头文件里主要是声明界面用的元素,及一些槽函数。重点是声明udpSocket。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2
3 MainWindow::MainWindow(QWidget *parent)
4 : QMainWindow(parent)
5 {
6
7 this->setGeometry(0, 0, 800, 480);
8
9
10 udpSocket = new QUdpSocket(this);
11
12
13 pushButton[0] = new QPushButton();
14
15 pushButton[1] = new QPushButton();
16
17 pushButton[2] = new QPushButton();
18
19 pushButton[3] = new QPushButton();
20
21 pushButton[4] = new QPushButton();
22
23
24 hBoxLayout[0] = new QHBoxLayout();
25
26 hBoxLayout[1] = new QHBoxLayout();
27
28 hBoxLayout[2] = new QHBoxLayout();
29
30 hBoxLayout[3] = new QHBoxLayout();
31
32
33 hWidget[0] = new QWidget();
34
35 hWidget[1] = new QWidget();
36
37 hWidget[2] = new QWidget();
38
39
40 vWidget = new QWidget();
41 vBoxLayout = new QVBoxLayout();
42
43
44 label[0] = new QLabel();
45 label[1] = new QLabel();
46 label[2] = new QLabel();
47
48 lineEdit = new QLineEdit();
49 comboBox = new QComboBox();
50 spinBox[0] = new QSpinBox();
51 spinBox[1] = new QSpinBox();
52 textBrowser = new QTextBrowser();
53
54 label[0]->setText("目标IP地址:");
55 label[1]->setText("绑定端口:");
56 label[2]->setText("目标端口:");
57
58
59 label[0]->setSizePolicy(QSizePolicy::Fixed,
60 QSizePolicy::Fixed);
61 label[1]->setSizePolicy(QSizePolicy::Fixed,
62 QSizePolicy::Fixed);
63 label[2]->setSizePolicy(QSizePolicy::Fixed,
64 QSizePolicy::Fixed);
65
66
67 spinBox[0]->setRange(10000, 99999);
68 spinBox[1]->setRange(10000, 99999);
69
70 pushButton[0]->setText("绑定端口");
71 pushButton[1]->setText("解除绑定");
72 pushButton[2]->setText("清空文本");
73 pushButton[3]->setText("发送消息");
74 pushButton[4]->setText("广播消息");
75
76
77 pushButton[1]->setEnabled(false);
78
79
80 lineEdit->setText("您好!");
81
82
83 hBoxLayout[0]->addWidget(pushButton[0]);
84 hBoxLayout[0]->addWidget(pushButton[1]);
85 hBoxLayout[0]->addWidget(pushButton[2]);
86
87
88 hWidget[0]->setLayout(hBoxLayout[0]);
89
90 hBoxLayout[1]->addWidget(label[0]);
91 hBoxLayout[1]->addWidget(comboBox);
92 hBoxLayout[1]->addWidget(label[1]);
93 hBoxLayout[1]->addWidget(spinBox[0]);
94 hBoxLayout[1]->addWidget(label[2]);
95 hBoxLayout[1]->addWidget(spinBox[1]);
96
97
98 hWidget[1]->setLayout(hBoxLayout[1]);
99
100
101 hBoxLayout[2]->addWidget(lineEdit);
102 hBoxLayout[2]->addWidget(pushButton[3]);
103 hBoxLayout[2]->addWidget(pushButton[4]);
104
105
106 hWidget[2]->setLayout(hBoxLayout[2]);
107
108
109 vBoxLayout->addWidget(textBrowser);
110 vBoxLayout->addWidget(hWidget[1]);
111 vBoxLayout->addWidget(hWidget[0]);
112 vBoxLayout->addWidget(hWidget[2]);
113
114
115 vWidget->setLayout(vBoxLayout);
116
117
118 setCentralWidget(vWidget);
119
120
121 getLocalHostIP();
122
123
124 connect(pushButton[0], SIGNAL(clicked()),
125 this, SLOT(bindPort()));
126 connect(pushButton[1], SIGNAL(clicked()),
127 this, SLOT(unbindPort()));
128 connect(pushButton[2], SIGNAL(clicked()),
129 this, SLOT(clearTextBrowser()));
130 connect(pushButton[3], SIGNAL(clicked()),
131 this, SLOT(sendMessages()));
132 connect(pushButton[4], SIGNAL(clicked()),
133 this, SLOT(sendBroadcastMessages()));
134 connect(udpSocket, SIGNAL(readyRead()),
135 this, SLOT(receiveMessages()));
136 connect(udpSocket,
137 SIGNAL(stateChanged(QAbstractSocket::SocketState)),
138 this,
139 SLOT(socketStateChange(QAbstractSocket::SocketState)));
140 }
141
142 MainWindow::~MainWindow()
143 {
144 }
145
146 void MainWindow::bindPort()
147 {
148 quint16 port = spinBox[0]->value();
149
150
151 if (udpSocket->state() != QAbstractSocket::UnconnectedState)
152 udpSocket->close();
153
154 if (udpSocket->bind(port)) {
155 textBrowser->append("已经成功绑定端口:"
156 + QString::number(port));
157
158
159 pushButton[0]->setEnabled(false);
160 pushButton[1]->setEnabled(true);
161 spinBox[1]->setEnabled(false);
162 }
163 }
164
165 void MainWindow::unbindPort()
166 {
167
168 udpSocket->abort();
169
170
171 pushButton[0]->setEnabled(true);
172 pushButton[1]->setEnabled(false);
173 spinBox[1]->setEnabled(true);
174 }
175
176
177 void MainWindow::getLocalHostIP()
178 {
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
198 QList<QNetworkInterface> list
199 = QNetworkInterface::allInterfaces();
200
201
202 foreach (QNetworkInterface interface, list) {
203
204
205 QList<QNetworkAddressEntry> entryList
206 = interface.addressEntries();
207
208
209 foreach (QNetworkAddressEntry entry, entryList) {
210
211 if (entry.ip().protocol() ==
212 QAbstractSocket::IPv4Protocol) {
213 comboBox->addItem(entry.ip().toString());
214
215 IPlist<<entry.ip();
216 }
217 }
218 }
219 }
220
221
222 void MainWindow::clearTextBrowser()
223 {
224
225 textBrowser->clear();
226 }
227
228
229 void MainWindow::receiveMessages()
230 {
231
232 QHostAddress peerAddr;
233 quint16 peerPort;
234
235
236 while (udpSocket->hasPendingDatagrams()) {
237
238 QByteArray datagram;
239
240
241 datagram.resize(udpSocket->pendingDatagramSize());
242
243
244 udpSocket->readDatagram(datagram.data(),
245 datagram.size(),
246 &peerAddr,
247 &peerPort);
248
249 QString str = datagram.data();
250
251
252 textBrowser->append("接收来自"
253 + peerAddr.toString()
254 + ":"
255 + QString::number(peerPort)
256 + str);
257 }
258 }
259
260
261 void MainWindow::sendMessages()
262 {
263
264 textBrowser->append("发送:" + lineEdit->text());
265
266
267 QByteArray data = lineEdit->text().toUtf8();
268
269
270 QHostAddress peerAddr = IPlist[comboBox->currentIndex()];
271
272
273 quint16 peerPort = spinBox[1]->value();
274
275
276 udpSocket->writeDatagram(data, peerAddr, peerPort);
277 }
278
279 void MainWindow::sendBroadcastMessages()
280 {
281
282 textBrowser->append("发送:" + lineEdit->text());
283
284
285 QByteArray data = lineEdit->text().toUtf8();
286
287
289 QHostAddress peerAddr = QHostAddress::Broadcast;
290
291
292 quint16 peerPort = spinBox[1]->text().toInt();
293
294
295 udpSocket->writeDatagram(data, peerAddr, peerPort);
296 }
297
298 void MainWindow::socketStateChange(QAbstractSocket::SocketState state)
299 {
300 switch (state) {
301 case QAbstractSocket::UnconnectedState:
302 textBrowser->append("scoket状态:UnconnectedState");
303 break;
304 case QAbstractSocket::ConnectedState:
305 textBrowser->append("scoket状态:ConnectedState");
306 break;
307 case QAbstractSocket::ConnectingState:
308 textBrowser->append("scoket状态:ConnectingState");
309 break;
310 case QAbstractSocket::HostLookupState:
311 textBrowser->append("scoket状态:HostLookupState");
312 break;
313 case QAbstractSocket::ClosingState:
314 textBrowser->append("scoket状态:ClosingState");
315 break;
316 case QAbstractSocket::ListeningState:
317 textBrowser->append("scoket状态:ListeningState");
318 break;
319 case QAbstractSocket::BoundState:
320 textBrowser->append("scoket状态:BoundState");
321 break;
322 default:
323 break;
324 }
325 }
第146~163行,绑定端口。使用bind方法,即可绑定一个端口。注意我们绑定的端口不能和主机已经使用的端口冲突! 第165~174行,解绑端口。使用abort方法即可解绑。 第229~258行,接收消息,注意接收消息是QByteArray字节数组。读数组使用的是readDatagram方法,在readDatagram方法里可以获取对方的套接字IP地址与端口号。 第261~277行,单播消息,需要知道目标IP与目标端口号。即可用writeDatagram方法发送消息。 第279~296行,广播消息与单播消息不同的是将目标IP地址换成了广播地址,一般广播地址为255.255.255.255。 11.3.2.2 程序运行效果 本实例可以做即是发送者,也是接收者。如果在同一台主机同一个系统里运行两个本例程序。不能绑定同一个端口!否则会冲突!当您想测试在同一局域网内不同主机上运行此程序,那么绑定的端口号可以相同。 本例设置目标IP地址为127.0.0.1,此IP地址是Ubuntu/Windows上的环回IP地址,可以用于无网络时测试。绑定端口号与目标端口号相同,也就是说,此程序正在监听端口号为10000的数据,此程序也向目标IP地址127.0.0.1的10000端口号发送数据,实际上此程序就完成了自发自收。 当我们点击发送消息按钮时,文本消息窗口显示发送的数据“您好!”,同时接收到由本地IP 127.0.0.1发出的数据“您好!”。其中ffff:是通信套接字的标识。呵呵!您可能会问为什么不是本主机的其它地址如(192.168.1.x)发出的呢?因为我们选择了目标的IP地址为127.0.0.1,那么要与此目标地址通信,必须使用相同网段的IP设备与之通信。注意不能用本地环回发送消息到其他主机上。因为本地环回IP只适用于本地主机上的IP通信。 当我们点击广播消息按钮时,广播发送的目标IP地址变成了广播地址255.255.255.255。那么我们将收到从本地IP地址192.168.x.x的数据。如下图,收到了从192.168.1.129发送过来的数据。因为环回IP 127.0.0.1的广播地址为255.0.0.0,所以要与255.255.255.255的网段里的IP通信数据必须是由192.168.x.x上发出的。如果其他同一网段上的其他主机正在监听目标端口,那么它们将同时收到消息。这也验证了上一小节为什么会从127.0.0.1发送数据。 本例不难,可能有点绕,大家多参考资料理解理解,知识点有点多,如果没有些通信基础的话,我们需要慢慢吃透。
11.3.3 UDP组播 通常,在传统的网络通讯中,有两种方式,一种是源主机和目标主机两台主机之间进行的“一对一”的通讯方式,即单播,第二种是一台源主机与网络中所有其他主机之间进行的通讯,即广播。那么,如果需要将信息从源主机发送到网络中的多个目标主机,要么采用广播方式,这样网络中所有主机都会收到信息,要么,采用单播方式,由源主机分别向各个不同目标主机发送信息。可以看出来,在广播方式下,信息会发送到不需要该信息的主机从而浪费带宽资源,甚至引起广播风暴:而单播方式下,会因为数据包的多次重复而浪费带宽资源,同时,源主机的负荷会因为多次的数据复制而加大,所以,单播与广播对于多点发送问题有缺陷。在此情况下,组播技术就应用而生了。 组播类似于QQ群,如果把腾讯向QQ每个用户发送推送消息比作广播,那么组播就像是QQ群一样,只有群内的用户才能收到消息。想要收到消息,我们得先加群。 一个D类IP地址的第一个字节必须以“1110”开始,D类IP地址不分网络地址和主机地址,是一个专门保留的地址,其地址范围为224.0.0.0~239.255.255.255。D类IP地址主要用于多点广播(Multicast,也称为多播(组播))之中作为多播组IP地址。其中,多播组IP地址让源主机能够将分组发送给网络中的一组主机,属于多播组的主机将被分配一个多播组lP地址。由于多播组lP地址标识了一组主机(也称为主机组),因此多播组IP地址只能作为目标地址,源地址总是为单播地址。 ?224.0.0.0~224.0.0.255为预留的组播地址(永久组地址),地址224.0.0.0保留不做分配,其它地址供路由协议使用。 ?224.0.1.0~238.255.255.255为用户可用的组播地址(临时组地址),全网范围内有效。 ?239.0.0.0~239.255.255.255为本地管理组播地址,仅在特定的本地范围内有效。 通过以上的信息,我们只需要关注,哪些组播地址可以被我们在本地主机使用即可。在家庭网络和办公网络局域网内使用UDP组播功能,那么可用的组播地址范围是239.0.0.0~239.255.255.255。 QUdpSocket类支持UDP组播,提供了joinMulticastGroup方法使本地主机加入多播组,leaveMulticastGroup离开多播组。其他绑定端口,发送接收功能与UDP单播和广播完全一样。实际上我们在上一个实例学会使用joinMulticastGroup和leaveMulticastGroup的应用即可! 11.3.3.1 应用实例 本例目的:了解QUdpSocket组播使用。 例11_udp_multicast,UDP单播与广播应用(难度:一般)。项目路径为Qt/2/11_udp_multicast。本例大体流程首先获取本地IP地址。创建一个udpSocket套接字,加入组播前必须绑定本机主机的端口。加入组播使用joinMulticastGroup,退出组播使用leaveMulticastGroup。其他收发消息的功能与上一节单播和广播一样。 项目文件10_udp_unicast_broadcast.pro文件第一行添加的代码部分如下。 11_udp_multicast.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QUdpSocket>
6 #include <QVBoxLayout>
7 #include <QHBoxLayout>
8 #include <QPushButton>
9 #include <QTextBrowser>
10 #include <QLabel>
11 #include <QComboBox>
12 #include <QSpinBox>
13 #include <QHostInfo>
14 #include <QLineEdit>
15 #include <QNetworkInterface>
16 #include <QDebug>
17
18 class MainWindow : public QMainWindow
19 {
20 Q_OBJECT
21
22 public:
23 MainWindow(QWidget *parent = nullptr);
24 ~MainWindow();
25
26 private:
27
28 QUdpSocket *udpSocket;
29
30
31 QPushButton *pushButton[4];
32
33
34 QLabel *label[3];
35
36
37 QWidget *hWidget[3];
38
39
40 QHBoxLayout *hBoxLayout[3];
41
42
43 QWidget *vWidget;
44
45
46 QVBoxLayout *vBoxLayout;
47
48
49 QTextBrowser *textBrowser;
50
51
52 QComboBox *comboBox[2];
53
54
55 QSpinBox *spinBox;
56
57
58 QLineEdit *lineEdit;
59
60
61 QList<QHostAddress> IPlist;
62
63
64 void getLocalHostIP();
65
66 private slots:
67
68 void joinGroup();
69
70
71 void leaveGroup();
72
73
74 void clearTextBrowser();
75
76
77 void receiveMessages();
78
79
80 void sendMessages();
81
82
83 void socketStateChange(QAbstractSocket::SocketState);
84 };
85 #endif
头文件里主要是声明界面用的元素,及一些槽函数。重点是声明udpSocket。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2
3 MainWindow::MainWindow(QWidget *parent)
4 : QMainWindow(parent)
5 {
6
7 this->setGeometry(0, 0, 800, 480);
8
9
10 udpSocket = new QUdpSocket(this);
11
12
14 udpSocket->setSocketOption(QAbstractSocket::MulticastTtlOption, 1);
15
16
17 pushButton[0] = new QPushButton();
18
19 pushButton[1] = new QPushButton();
20
21 pushButton[2] = new QPushButton();
22
23 pushButton[3] = new QPushButton();
24
25
26 hBoxLayout[0] = new QHBoxLayout();
27
28 hBoxLayout[1] = new QHBoxLayout();
29
30 hBoxLayout[2] = new QHBoxLayout();
31
32 hBoxLayout[3] = new QHBoxLayout();
33
34
35 hWidget[0] = new QWidget();
36
37 hWidget[1] = new QWidget();
38
39 hWidget[2] = new QWidget();
40
41
42 vWidget = new QWidget();
43 vBoxLayout = new QVBoxLayout();
44
45
46 label[0] = new QLabel();
47 label[1] = new QLabel();
48 label[2] = new QLabel();
49
50 lineEdit = new QLineEdit();
51 comboBox[0] = new QComboBox();
52 comboBox[1] = new QComboBox();
53 spinBox = new QSpinBox();
54 textBrowser = new QTextBrowser();
55
56 label[0]->setText("本地IP地址:");
57 label[1]->setText("组播地址:");
58 label[2]->setText("组播端口:");
59
60
61 label[0]->setSizePolicy(QSizePolicy::Fixed,
62 QSizePolicy::Fixed);
63 label[1]->setSizePolicy(QSizePolicy::Fixed,
64 QSizePolicy::Fixed);
65 label[2]->setSizePolicy(QSizePolicy::Fixed,
66 QSizePolicy::Fixed);
67
68
69 spinBox->setRange(10000, 99999);
70
71 pushButton[0]->setText("加入组播");
72 pushButton[1]->setText("退出组播");
73 pushButton[2]->setText("清空文本");
74 pushButton[3]->setText("组播消息");
75
76
77 pushButton[1]->setEnabled(false);
78
79
80 lineEdit->setText("您好!");
81
82
83 comboBox[1]->addItem("239.255.255.1");
84
85
86 comboBox[1]->setEditable(true);
87
88
89 hBoxLayout[0]->addWidget(pushButton[0]);
90 hBoxLayout[0]->addWidget(pushButton[1]);
91 hBoxLayout[0]->addWidget(pushButton[2]);
92
93
94 hWidget[0]->setLayout(hBoxLayout[0]);
95
96 hBoxLayout[1]->addWidget(label[0]);
97 hBoxLayout[1]->addWidget(comboBox[0]);
98 hBoxLayout[1]->addWidget(label[1]);
99 hBoxLayout[1]->addWidget(comboBox[1]);
100 hBoxLayout[1]->addWidget(label[2]);
101 hBoxLayout[1]->addWidget(spinBox);
102
103
104 hWidget[1]->setLayout(hBoxLayout[1]);
105
106
107 hBoxLayout[2]->addWidget(lineEdit);
108 hBoxLayout[2]->addWidget(pushButton[3]);
109
110
111 hWidget[2]->setLayout(hBoxLayout[2]);
112
113
114 vBoxLayout->addWidget(textBrowser);
115 vBoxLayout->addWidget(hWidget[1]);
116 vBoxLayout->addWidget(hWidget[0]);
117 vBoxLayout->addWidget(hWidget[2]);
118
119
120 vWidget->setLayout(vBoxLayout);
121
122
123 setCentralWidget(vWidget);
124
125
126 getLocalHostIP();
127
128
129 connect(pushButton[0], SIGNAL(clicked()),
130 this, SLOT(joinGroup()));
131 connect(pushButton[1], SIGNAL(clicked()),
132 this, SLOT(leaveGroup()));
133 connect(pushButton[2], SIGNAL(clicked()),
134 this, SLOT(clearTextBrowser()));
135 connect(pushButton[3], SIGNAL(clicked()),
136 this, SLOT(sendMessages()));
137 connect(udpSocket, SIGNAL(readyRead()),
138 this, SLOT(receiveMessages()));
139 connect(udpSocket,
140 SIGNAL(stateChanged(QAbstractSocket::SocketState)),
141 this,
142 SLOT(socketStateChange(QAbstractSocket::SocketState)));
143 }
144
145 MainWindow::~MainWindow()
146 {
147 }
148
149 void MainWindow::joinGroup()
150 {
151
152 quint16 port = spinBox->value();
153
154 QHostAddress groupAddr = QHostAddress(comboBox[1]->currentText());
155
156
157 if (udpSocket->state() != QAbstractSocket::UnconnectedState)
158 udpSocket->close();
159
160
161 if (udpSocket->bind(QHostAddress::AnyIPv4,
162 port, QUdpSocket::ShareAddress)) {
163
164
165 bool ok = udpSocket->joinMulticastGroup(groupAddr);
166
167 textBrowser->append(ok ? "加入组播成功" : "加入组播失败");
168
169 textBrowser->append("组播地址IP:"
170 + comboBox[1]->currentText());
171
172 textBrowser->append("绑定端口:"
173 + QString::number(port));
174
175
176 pushButton[0]->setEnabled(false);
177 pushButton[1]->setEnabled(true);
178 comboBox[1]->setEnabled(false);
179 spinBox->setEnabled(false);
180 }
181 }
182
183 void MainWindow::leaveGroup()
184 {
185
186 QHostAddress groupAddr = QHostAddress(comboBox[1]->currentText());
187
188
189 udpSocket->leaveMulticastGroup(groupAddr);
190
191
192 udpSocket->abort();
193
194
195 pushButton[0]->setEnabled(true);
196 pushButton[1]->setEnabled(false);
197 comboBox[1]->setEnabled(true);
198 spinBox->setEnabled(true);
199 }
200
201
202 void MainWindow::getLocalHostIP()
203 {
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
223 QList<QNetworkInterface> list
224 = QNetworkInterface::allInterfaces();
225
226
227 foreach (QNetworkInterface interface, list) {
228
229
230 QList<QNetworkAddressEntry> entryList
231 = interface.addressEntries();
232
233
234 foreach (QNetworkAddressEntry entry, entryList) {
235
236 if (entry.ip().protocol() ==
237 QAbstractSocket::IPv4Protocol &&
238 ! entry.ip().isLoopback()) {
239
240 comboBox[0]->addItem(entry.ip().toString());
241
242 IPlist<<entry.ip();
243 }
244 }
245 }
246 }
247
248
249 void MainWindow::clearTextBrowser()
250 {
251
252 textBrowser->clear();
253 }
254
255
256 void MainWindow::receiveMessages()
257 {
258
259 QHostAddress peerAddr;
260 quint16 peerPort;
261
262
263 while (udpSocket->hasPendingDatagrams()) {
264
265 QByteArray datagram;
266
267
268 datagram.resize(udpSocket->pendingDatagramSize());
269
270
271 udpSocket->readDatagram(datagram.data(),
272 datagram.size(),
273 &peerAddr,
274 &peerPort);
275
276 QString str = datagram.data();
277
278
279 textBrowser->append("接收来自"
280 + peerAddr.toString()
281 + ":"
282 + QString::number(peerPort)
283 + str);
284 }
285 }
286
287
288 void MainWindow::sendMessages()
289 {
290
291 textBrowser->append("发送:" + lineEdit->text());
292
293
294 QByteArray data = lineEdit->text().toUtf8();
295
296
297 QHostAddress groupAddr = QHostAddress(comboBox[1]->currentText());
298
299
300 quint16 groupPort = spinBox->value();
301
302
303 udpSocket->writeDatagram(data, groupAddr, groupPort);
304 }
305
306
307 void MainWindow::socketStateChange(QAbstractSocket::SocketState state)
308 {
309 switch (state) {
310 case QAbstractSocket::UnconnectedState:
311 textBrowser->append("scoket状态:UnconnectedState");
312 break;
313 case QAbstractSocket::ConnectedState:
314 textBrowser->append("scoket状态:ConnectedState");
315 break;
316 case QAbstractSocket::ConnectingState:
317 textBrowser->append("scoket状态:ConnectingState");
318 break;
319 case QAbstractSocket::HostLookupState:
320 textBrowser->append("scoket状态:HostLookupState");
321 break;
322 case QAbstractSocket::ClosingState:
323 textBrowser->append("scoket状态:ClosingState");
324 break;
325 case QAbstractSocket::ListeningState:
326 textBrowser->append("scoket状态:ListeningState");
327 break;
328 case QAbstractSocket::BoundState:
329 textBrowser->append("scoket状态:BoundState");
330 break;
331 default:
332 break;
333 }
334 }
第161~162行,绑定端口。使用bind方法,即可绑定一个端口。注意我们绑定的端口不能和主机已经使用的端口冲突! 第165行,使用joinMulticastGroup加入组播,QHostAddress::AnyIPv4,是加入Ipv4组播的一个接口,所有操作系统都不支持不带接口选择的加入IPv6组播组。加入的结果返回给变量ok。组播地址可由用户点击comboBox[1]控件输入(默认编者已经输入一个地址为239.255.255.1),注意组播地址的范围必须是239.0.0.0~239.255.255.255中的一个数。 第189行,使用leaveMulticastGroup退出组播。 第192行,解绑端口。使用abort方法即可解绑。 第256~285行,接收消息,注意接收消息是QByteArray字节数组。读数组使用的是readDatagram方法,在readDatagram方法里可以获取对方的套接字IP地址与端口号。 第288~304行,发送消息,组播与广播消息或单播消息不同的是将目标IP地址换成了组播地址239.255.255.1。 11.3.3.2 程序运行效果 运行程序后,点击加入组播,然后点击组播消息,本实例可以做即是发送者,也是接收者。如果在同一台主机同一个系统里运行两个本例程序。不能绑定同一个端口!否则会冲突!当您想测试在同一局域网内不同主机上运行此程序,那么绑定的端口号可以相同。 因为是组播消息,所以自己也会收到消息,如果在局域网内其他主机运行此程序,当点击加入组播后,就可以收发消息了。
11.4 网络下载实例 Qt网络模块还提供了直接访问如HTTP,FTP等网络协议的类,这些类是QNetworkAccessManager、QNetworkRequest和QNetworkReply。 通常需要这三个类协作才能完成一个网络操作。可以用于从网络获取时间,天气和图片等等数据。比如本例需要下载一张图片,大概流程如下。 由QNetworkRequest类设置一个URL地址发起网络协议请求,QNetworkRequest类保存要用QNetworkAccessManager发送的请求。QNetworkRequest是网络访问API的一部分,是一个持有通过网络发送请求所需信息的类。它包含一个URL和一些可用于修改请求的辅助信息。 QNetworkAccessManager类允许应用程序发送网络请求并接收响应。在QNetworkRequest发起网络请求后,QNetworkAccessManager负责发送网络请求,创建网络响应。 QNetworkReply类就用于QNetworkAccessManager创建的网络响应。最终由QNetworkReply处理网络响应。它提供了finished()、readyRead()和downloadProgress()等信号,可以监测网络响应的执行情况。并且QNetworkReply继承于QIODevice,所以QNetworkReply支持流读写,可以直接用read()和write等功能。 11.4.1 应用实例 本例目的:了解QNetworkAccessManager、QNetworkRequest和QNetworkReply类的使用。 例12_imagedownload,下载小图片(难度:一般)。项目路径为Qt/2/12_imagedownload。本例大体流程,设置一个下载图片的URL,通过networkReply处理响应后,从流中读取图片的数据,然后保存到本地。 项目文件12_imagedownload.pro文件第一行添加的代码部分如下。 12_imagedownload.pro编程后的代码
1 QT += core gui network
2
3 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
4
5 CONFIG += c++11
6
7 # The following define makes your compiler emit warnings if you use
8 # any Qt feature that has been marked deprecated (the exact warnings
9 # depend on your compiler). Please consult the documentation of the
10 # deprecated API in order to know how to port your code away from it.
11 DEFINES += QT_DEPRECATED_WARNINGS
12
13 # You can also make your code fail to compile if it uses deprecated APIs.
14 # In order to do so, uncomment the following line.
15 # You can also select to disable deprecated APIs only up to a certain version of Qt.
16 #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
17
18 SOURCES += \
19 main.cpp \
20 mainwindow.cpp
21
22 HEADERS += \
23 mainwindow.h
24
25 # Default rules for deployment.
26 qnx: target.path = /tmp/$${TARGET}/bin
27 else: unix:!android: target.path = /opt/$${TARGET}/bin
28 !isEmpty(target.path): INSTALLS += target
在头文件“mainwindow.h”具体代码如下。 mainwindow.h编程后的代码
1 #ifndef MAINWINDOW_H
2 #define MAINWINDOW_H
3
4 #include <QMainWindow>
5 #include <QNetworkAccessManager>
6 #include <QNetworkReply>
7 #include <QFile>
8 #include <QLabel>
9 #include <QPushButton>
10 #include <QProgressBar>
11 #include <QHBoxLayout>
12 #include <QVBoxLayout>
13 #include <QLineEdit>
14
15 class MainWindow : public QMainWindow
16 {
17 Q_OBJECT
18
19 public:
20 MainWindow(QWidget *parent = nullptr);
21 ~MainWindow();
22 private:
23
24 QNetworkAccessManager *networkAccessManager;
25
26
27 QLabel *label[3];
28
29
30 QPushButton *pushButton;
31
32
33 QProgressBar *progressBar;
34
35
36 QHBoxLayout *hBoxLayout[2];
37
38
39 QVBoxLayout *vBoxLayout;
40
41
42 QWidget *hWidget[2];
43
44
45 QWidget *vWidget;
46
47
48 QLineEdit *lineEdit;
49
50 private slots:
51
52 void readyReadData();
53
54
55 void replyFinished();
56
57
58 void imageDownloadProgress(qint64, qint64);
59
60
61 void startDownload();
62
63
64 void networkReplyError(QNetworkReply::NetworkError);
65 };
66 #endif // MAINWINDOW_H 头文件里主要是声明界面用的元素,及一些槽函数。重点是声明networkAccessManager。 在源文件“mainwindow.cpp”具体代码如下。 mainwindow.cpp编程后的代码
1 #include "mainwindow.h"
2 #include <QMessageBox>
3 #include <QCoreApplication>
4
5 MainWindow::MainWindow(QWidget *parent)
6 : QMainWindow(parent)
7 {
8
9 this->setGeometry(0, 0, 800, 480);
10
11
12 label[0] = new QLabel();
13
14 label[1] = new QLabel();
15
16 label[2] = new QLabel();
17
18
19 lineEdit = new QLineEdit();
20
21
22 pushButton = new QPushButton();
23
24
25 progressBar = new QProgressBar();
26
27
28 hBoxLayout[0] = new QHBoxLayout();
29 hBoxLayout[1] = new QHBoxLayout();
30
31
32 vBoxLayout = new QVBoxLayout();
33
34
35 hWidget[0] = new QWidget();
36 hWidget[1] = new QWidget();
37
38
39 vWidget = new QWidget();
40
41 label[1]->setText("URL链接:");
42 label[2]->setText("文件下载进度:");
43
44 pushButton->setText("下载");
45
46
47 lineEdit->setText("https://ss0.bdstatic.com/70cFuH"
48 "Sh_Q1YnxGkpoWK1HF6hhy/it/u=42710"
49 "87328,1384669424&fm=11&gp=0.jpg");
50
51 label[0]->setMinimumSize(this->width(),
52 this->height() * 0.75);
53
54
55 label[1]->setSizePolicy(QSizePolicy::Fixed,
56 QSizePolicy::Fixed);
57 label[2]->setSizePolicy(QSizePolicy::Fixed,
58 QSizePolicy::Fixed);
59 pushButton->setSizePolicy(QSizePolicy::Fixed,
60 QSizePolicy::Fixed);
61
62
63 hBoxLayout[0]->addWidget(label[1]);
64 hBoxLayout[0]->addWidget(lineEdit);
65 hBoxLayout[0]->addWidget(pushButton);
66
67
68 hWidget[0]->setLayout(hBoxLayout[0]);
69
70
71 hBoxLayout[1]->addWidget(label[2]);
72 hBoxLayout[1]->addWidget(progressBar);
73
74
75 hWidget[1]->setLayout(hBoxLayout[1]);
76
77
78 vBoxLayout->addWidget(label[0]);
79 vBoxLayout->addWidget(hWidget[0]);
80 vBoxLayout->addWidget(hWidget[1]);
81
82
83 vWidget->setLayout(vBoxLayout);
84
85
86 setCentralWidget(vWidget);
87
88
89 networkAccessManager = new QNetworkAccessManager(this);
90
91
92 connect(pushButton, SIGNAL(clicked()),
93 this, SLOT(startDownload()));
94
95 }
96
97 MainWindow::~MainWindow()
98 {
99 }
100
101 void MainWindow::startDownload()
102 {
103
104 QUrl newUrl(QUrl(lineEdit->text()));
105
106
107 if (!newUrl.isValid()) {
108 QMessageBox::information(this, "error", "invalid url");
109 return;
110 }
111
112
113 QNetworkRequest networkRequest;
114
115
116 networkRequest.setUrl(newUrl);
117
118
119 QNetworkReply *newReply =
120 networkAccessManager->get(networkRequest);
121
122
123 connect(newReply, SIGNAL(finished()),
124 this, SLOT(replyFinished()));
125 connect(newReply, SIGNAL(readyRead()),
126 this, SLOT(readyReadData()));
127 connect(newReply, SIGNAL(downloadProgress(qint64, qint64)),
128 this, SLOT(imageDownloadProgress(qint64, qint64)));
129 connect(newReply,
130 SIGNAL(error(QNetworkReply::NetworkError)),
131 this,
132 SLOT(networkReplyError(QNetworkReply::NetworkError )));
133 }
134
135 void MainWindow::readyReadData()
136 {
137
138 pushButton->setEnabled(false);
139
140
141 QNetworkReply *reply = (QNetworkReply *)sender();
142
143 QFile imageFile;
144
145 imageFile.setFileName(QCoreApplication::applicationDirPath()
146 + "/下载的.jpg");
147
148
149 if (imageFile.exists())
150 imageFile.remove();
151
152
153 QByteArray data = reply->readAll();
154
155 if (data.isEmpty()) {
156 qDebug()<<"data is null, please try it again!"<<endl;
157 return;
158 }
159
160
161 if (! (data[0] == (char)0xff
162 && data[1] == (char)0xd8
163 && data[data.size() - 2] == (char)0xff
164 && data[data.size() - 1] == (char)0xd9)) {
165 qDebug()<<"not JPG data, please try it again!"<<endl;
166 return;
167 }
168
169
170 QPixmap pixmap;
171 pixmap.loadFromData(data);
172 pixmap.save(imageFile.fileName());
173 }
174
175 void MainWindow::replyFinished()
176 {
177
178 QNetworkReply *reply = (QNetworkReply *)sender();
179
180
181 reply->deleteLater();
182
183
184 QFile imageFile(QCoreApplication::applicationDirPath()
185 + "/下载的.jpg");
186 if (imageFile.exists()) {
187
188 label[0]->setPixmap(QPixmap(imageFile.fileName()));
189 qDebug() <<"已经成功下载,文件路径为:"
190 <<imageFile.fileName()<<endl;
191 } else
192
193 label[0]->clear();
194
195
196 pushButton->setEnabled(true);
197 }
198
199 void MainWindow::imageDownloadProgress(qint64 bytes,
200 qint64 totalBytes)
201 {
202
203 progressBar->setMaximum(totalBytes);
204
205 progressBar->setValue(bytes);
206 }
207
208
209 void MainWindow::networkReplyError(QNetworkReply::NetworkError
210 error)
211 {
212 switch (error) {
213 case QNetworkReply::ConnectionRefusedError:
214 qDebug()<<"远程服务器拒绝连接"<<endl;
215 break;
216 case QNetworkReply::HostNotFoundError:
217 qDebug()<<"找不到远程主机名"<<endl;
218 break;
219 case QNetworkReply::TimeoutError:
220 qDebug()<<"与远程服务器连接超时"<<endl;
221 break;
222 default:
223 break;
224 }
225 }
第89行,全局变量networkAccessManager实例化。 第101行~133行,首先从单行输入框里获取URL链接为newUrl,判断链接的有效性。然后创建局部变量networkRequest,设置networkRequest请求URL为newUrl。QNetworkReply *newReply = networkAccessManager->get(networkRequest);为最重要的代码,所有响应本次的操作都交给了newReply。通过信号槽处理对应的操作。 第135~173行,这部分代码就是从newReply流里读取网络下载的数据。 第160~167行,这里编者做了一些处理,从网络下载的数据可能遇到数据丢失或者下载错误的情况。本例是从百度里下载一张JPG格式的图片,因为JPG图片的判断依据是第一个字节和第二个字节的数据是0xff和0xd8,倒数第一个和倒数第二个字节数据分别是0xd9和0xff。如果都对,那么判断此数据为JPG图片数据。然后进行保存,否则则不保存图片。处理数据往往是需要的,我们经常要对下载的数据进行处理。 第174~197行,网络响应完成。记得要删除reply,防止内存泄漏。如果下载成功图片,则显示图片到label[0]上。 第209~225行,网络响应错误处理函数。
11.4.2 程序运行效果 点击下载按钮后,可以看到本次下载的图片已经保存到当前编译输出的路径下,名字叫“下载的.jpg”,并显示到界面上。由于文件下载的速度非常快,所以下载的进度条一下子就变成了100%。若想看见下载进度条下载进度缓慢一些,可以修改本例去下载其他文件,注意不要保存为jpg图片了。注意:此程序里的下载链接可能失效,请替换自己的图片链接。
|