網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
本文實(shí)例為大家分享了Qt實(shí)現(xiàn)TCP網(wǎng)絡(luò)編程的具體代碼,供大家參考,具體內(nèi)容如下
1.Qt中的TCP客戶端編程
Qt中的TCP客戶端編程:
對(duì)于Qt編程而言,網(wǎng)絡(luò)只是數(shù)據(jù)傳輸?shù)耐ǖ?/strong>
Qt提供了QTcpSocket類(封裝了TCP協(xié)議細(xì)節(jié))
將QTcpSocket的對(duì)象當(dāng)做黑盒使用,進(jìn)行數(shù)據(jù)收發(fā)
QTcpSocket的使用方式:
1.連接服務(wù)端主機(jī)(connectToHost())
2.發(fā)送數(shù)據(jù)/接受數(shù)據(jù)(write()/read())
3.關(guān)閉連接(close())
QTcpSocket的注意事項(xiàng):
默認(rèn)情況下,QTcpSocket使用異步編程的方式:
操作完成后立即返回
通過(guò)發(fā)送信號(hào)的方式返回操作結(jié)果
QTcpSocket提供了輔助函數(shù),可完成同步編程的方式
waitForConnected()/waitForDisconnected()
waitForBytesWritten()/waitForReadyread()
QTcpSocket的同步編程:
編程實(shí)驗(yàn):同步編程
#include <QCoreApplication>
#include <QTcpSocket>
#include <QDebug>
#include <QThread>
void SyncClientDemo()
{
? ? QTcpSocket client;
? ? char buf[256] = {0};
? ? client.connectToHost("127.0.0.1",8080);
? ? qDebug() << "Connected:" << client.waitForConnected();
? ? qDebug() << "Send Bytes:" << client.write("CKY");
? ? qDebug() << "Send Status:" << client.waitForBytesWritten();
? ? qDebug() << "Data Avilable:" << client.waitForReadyRead();
? ? qDebug() << "Received Bytes:" << client.read(buf, sizeof(buf));
? ? qDebug() << "Received Data:" << buf;
? ? QThread::sleep(5000);
? ? client.close();
? ? client.waitForDisconnected();
}
int main(int argc, char *argv[])
{
? ? QCoreApplication a(argc, argv);
? ? SyncClientDemo();
? ? return a.exec();
}
QTcpSocket的異步編程:
QTcpSocket對(duì)象通過(guò)發(fā)送信號(hào)的方式返回操作結(jié)果
可以在程序中將對(duì)應(yīng)的信號(hào)連接到槽函數(shù),獲取結(jié)果
在GUI應(yīng)用程序中通常使用QTcpSocket的異步方式
QTcpSocket中的關(guān)鍵信號(hào):
connected():成功連接遠(yuǎn)端主機(jī)
disconnected():遠(yuǎn)端主機(jī)斷開連接
readyRead():遠(yuǎn)程數(shù)據(jù)到達(dá)本機(jī)
bytesWritten(qint64):數(shù)據(jù)成功發(fā)送至系統(tǒng)(OS)
編程實(shí)驗(yàn):QTcpSocket異步編程
#include "clientdemo.h"
#include <QDebug>
#include <QHostAddress>
ClientDemo::ClientDemo(QObject* parent) : QObject(parent)
{
? ? connect(&m_client, SIGNAL(connected()), this, SLOT(onConnected()));
? ? connect(&m_client, SIGNAL(disconnected()), this, SLOT(onDisconnected()));
? ? connect(&m_client, SIGNAL(readyRead()), this, SLOT(onDataReady()));
? ? connect(&m_client, SIGNAL(bytesWritten(qint64)), this, SLOT(onBytesWritten(qint64)));
}
void ClientDemo::onConnected()
{
? ? qDebug() << "onConnected()";
? ? qDebug() << "Local Address:" << m_client.localAddress();
? ? qDebug() << "Loacl Port:" << m_client.localPort();
}
void ClientDemo::onDisconnected()
{
? ? qDebug() << "onDiecennected()";
}
void ClientDemo::onDataReady()
{
? ? char buf[256] = {0};
? ? qDebug() << "onDataReady:" << m_client.read(buf, sizeof(buf));
? ? qDebug() << "Data:" << buf;
}
void ClientDemo::onBytesWritten(qint64 bytes)
{
? ? qDebug() << "onBytesWritten" << bytes;
}
void ClientDemo::connectTo(QString ip, int port)
{
? ? m_client.connectToHost(ip, port);
}
qint64 ClientDemo::send(const char* data, int len)
{
? ? return m_client.write(data, len);
}
qint64 ClientDemo::available()
{
? ? return m_client.bytesAvailable();
}
void ClientDemo::close()
{
? ? m_client.close();
}
1.Qt中的TCP服務(wù)端編程
網(wǎng)絡(luò)中的服務(wù)端:
服務(wù)端是為客戶端服務(wù)的,服務(wù)的內(nèi)容諸如向客戶端提供資源,保存客戶端數(shù)據(jù),為客戶端提供功能接口,等
Client/Server軟件架構(gòu)簡(jiǎn)介
特點(diǎn);
服務(wù)端被動(dòng)接受連接(服務(wù)端無(wú)法主動(dòng)連接客戶端)
服務(wù)端必須公開網(wǎng)絡(luò)地址(容易受到攻擊)
在職責(zé)上:
客戶端傾向于處理用于交互及體驗(yàn)(GUI)
服務(wù)端傾向于用戶數(shù)據(jù)的組織和存儲(chǔ)(數(shù)據(jù)處理)
B/S網(wǎng)絡(luò)結(jié)構(gòu)是什么?
Browser/Server軟件架構(gòu)簡(jiǎn)介
B/S是一種特殊的C/S網(wǎng)絡(luò)架構(gòu)
B/S中的客戶端統(tǒng)一使用瀏覽器(Browser)
B/S中的客戶端GUI通常采用HTML進(jìn)行開發(fā)
B/S中的客戶端與服務(wù)端通常采用http協(xié)議進(jìn)行通信
Qt中的TCP服務(wù)端編程:
Qt提供了QTcpServer類
將QTcpServer的對(duì)象當(dāng)做黑盒使用,進(jìn)行連接監(jiān)聽
每一個(gè)連接生成一個(gè)QTcpSocket對(duì)象進(jìn)行通信
QTcpServer的使用方式:
監(jiān)聽本機(jī)地址的端口(listen())
通過(guò)信號(hào)通知客戶端連接(newConnection())
獲取QTcpSocket通信對(duì)象(nextPendingConnection())
停止監(jiān)聽(close())
QTcpServer的注意事項(xiàng):
用于處理客戶端連接,不進(jìn)行具體通信
監(jiān)聽的端口只用于響應(yīng)連接請(qǐng)求
監(jiān)聽到連接后,生成QTcpSocket對(duì)象與客戶端通信
Client/Server交互流程:
編程實(shí)驗(yàn):QServerSocket編程
#include "serverdemo.h"
#include "QHostAddress"
#include <QDebug>
#include <QTcpServer>
#include <QObjectList>
ServerDemo::ServerDemo(QObject* parent) : QObject(parent)
{
? ? connect(&m_server, SIGNAL(newConnection()), this, SLOT(onNewConnection()));
}
void ServerDemo::onNewConnection()
{
? ? qDebug() << "onNewConnection";
? ? QTcpSocket* tcp = m_server.nextPendingConnection();
? ? connect(tcp, SIGNAL(connected()), this, SLOT(onConnected()));
? ? connect(tcp, SIGNAL(disconnected()), this, SLOT(onDisconnected()));
? ? connect(tcp, SIGNAL(readyRead()), this, SLOT(onDataReady()));
? ? connect(tcp, SIGNAL(bytesWritten(qint64)), this, SLOT(onBytesWritten(qint64)));
}
bool ServerDemo::start(int port)
{
? ? bool ret = true;
? ? if(!m_server.isListening())
? ? {
? ? ? ? ret = m_server.listen(QHostAddress("127.0.0.1", port));
? ? }
? ? return ret;
}
void ServerDemo::stop()
{
? ? if(m_server.isListening())
? ? {
? ? ? ? m_server.close();
? ? }
}
void ServerDemo::onConnected()
{
? ? QTcpServer* tcp = dynamic_cast<QTcpServer*>(sender());
? ? if(tcp != NULL)
? ? {
? ? ? ? qDebug() << "onConnected()";
? ? ? ? qDebug() << "Local Address:" << tcp->localAddress();
? ? ? ? qDebug() << "Loacl Port:" << tcp->localPort();
? ? }
}
void ServerDemo::onDisconnected()
{
? ? qDebug() << "onDiecennected()";
}
void ServerDemo::onDataReady()
{
? ? QTcpServer* tcp = dynamic_cast<QTcpServer*>(sender());
? ? char buf[256] = {0};
? ? if(tcp != NULL)
? ? {
? ? ? ? qDebug() << "onDataReady:" << tcp->read(buf, sizeof(buf));
? ? ? ? qDebug() << "Data:" << buf;
? ? }
}
void ServerDemo::onBytesWritten(qint64 bytes)
{
? ? qDebug() << "onBytesWritten" << bytes;
}
ServerDemo::~ServerDemo()
{
? ? const QObjectList& list = m_server.children();
? ? for(int i = 0; i < list.length(), i++)
? ? {
? ? ? ? QTcpSocket* tcp = dynamic_cast<QTcpSocket*>(list[i]);
? ? ? ? if(tcp != NULL)
? ? ? ? {
? ? ? ? ? ? tcp->close();
? ? ? ? }
? ? }
}
原文鏈接:https://blog.csdn.net/weixin_46571142/article/details/123029759
相關(guān)推薦
- 2024-01-08 AOP獲取方法返回值
- 2021-12-06 C語(yǔ)言使用rand函數(shù)生成隨機(jī)數(shù)_C 語(yǔ)言
- 2023-01-15 GoLang內(nèi)存泄漏原因排查詳解_Golang
- 2022-08-17 jQuery實(shí)現(xiàn)購(gòu)物車_jquery
- 2022-03-06 正則表達(dá)式用法詳解_正則表達(dá)式
- 2022-10-14 linux【centos 7】 yum 安裝 tesseract 4.1
- 2022-12-24 Kubernetes?kubectl中Pod創(chuàng)建流程源碼解析_云和虛擬化
- 2022-04-28 C#操作進(jìn)程的方法介紹_C#教程
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支