QT实现TCP/UDP通信

服务器端:

客户端:

服务器:

widget.h

cpp 复制代码
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpServer>
#include <QTcpSocket>
#include <QList>
#include <QMessageBox>
#include <QDebug>


QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

private slots:

    //自定义处理信号与槽函数
    void on_startButton_clicked();
    void newConnection_slot();
    void readyRead_slot();

private:
    Ui::Widget *ui;
    QTcpServer *server;//定义服务器指针
    QList<QTcpSocket*> socketList;//定义存放客户端信息的容器

};
#endif // WIDGET_H

widget.cpp

cpp 复制代码
#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    //实例化一个服务器对象
    server = new QTcpServer(this);//在堆区申请一个服务器


}

Widget::~Widget()
{
    delete ui;
}

//启动服务器按钮对应的槽函数
void Widget::on_startButton_clicked()
{
    if(ui->startButton->text()== "启动服务器")
    {
        //执行启动服务器动作

        //获取ui界面的端口号  ???
        quint16 port = ui->portEdit->text().toUInt();

        //启动监听
        //函数原型 bool listen(const QHostAddress &address = QHostAddress::any,quint16 port = 0);
        //1->要监听的ip地址 写any 表示监听该主机上所有的网络接口
        //2->要监听的端口号 不指定 系统默认随机绑定一个端口号

        if(!server->listen(QHostAddress::Any,port))
        {
            QMessageBox::critical(this,"错误","服务器启动失败");
            return;
        }

        //程序启动至此表示服务器启动成功
        QMessageBox::information(this,"成功","服务器启动成功");

        //将行编辑器设置为不可用
        ui->portEdit->setEnabled(false);
        //将按钮文本内容设置为关闭服务器
        ui->startButton->setText("关闭服务器");

        //此时 如果有客户端发来请求 那么该服务器会自动发射一个newConnection信号
        //将该信号连接到自定槽函数中处理后续操作
        connect(server,&QTcpServer::newConnection,this,&Widget::newConnection_slot);
    }
    else
    {
        //执行关闭服务器操作
        server->close();
        //将行编辑器设置成为可用状态
        ui->portEdit->setEnabled(true);
        //将按钮文本内容设置为启动服务器
        ui->startButton->setText("启动服务器");
    }
}

void Widget::newConnection_slot()
{
    qDebug()<<"有新的客户端发来连接请求";

    //可以通过nextPendingConnection函数获取最新连接的客户端套接字的地址
    //函数原型 QTcpSocket *nextPendingConnection();
    //无参函数
    //返回值 最新的连接的套接字地址

    QTcpSocket *s = server->nextPendingConnection();
    //将套接字地址放入链表中

    socketList.push_back(s);
    //一个服务器对应多个客户端 已经建立连接
    //如果有客户端发来消息 自动发射readyRead信号
    connect(s,&QTcpSocket::readyRead,this,&Widget::readyRead_slot);
}
void Widget::readyRead_slot()
{
    //遍历链表中的所有客户端 如果客户端状态未连接 则直接移除出链表
    for (int i=0 ;i<socketList.size();i++)
    {
        //判断当前套接字 socketList[i] 是否失效
        //函数原型 SocketState state() const;
        //返回值 返回当前调节子状态
        //返回结果为0 表示未连接
        if(socketList[i]->state()==0)
        {
            socketList.removeAt(i);
        }
    }

    //遍历所有客户端 判断客户端中是否有数据刻度 如果有数据可读 则表示该客户端发来的消息
    for (int i =0;i<socketList.count();i++)
    {
        //判断当前 客户端中是否有数据可读
        //函数原型 qint64 byteAvailable() const overrde
        //参数无
        //返回值为当前客户端套接字中的待读数据 如果没有数据 则返回0
        if(socketList[i]->bytesAvailable()  !=0)
        {
            //读取当前套接字的内容
            QByteArray msg = socketList[i]->readAll();

            //将接受的消息展示到ui界面上
            ui->msgListWidget->addItem(QString::fromLocal8Bit(msg));

            //将收到的消息 全部发给其他客户端
            for (int j=0;j<socketList.length();j++)
            {
                if (i!=j)
                {
                    socketList[j]->write(msg);
                }
            }
        }
    }
}

客户端

widget.h

cpp 复制代码
#ifndef WIDGET_H
#define WIDGET_H


#include <QWidget>
#include<QTcpSocket>         //客户端套接字类
#include<QMessageBox>        //消息对话框类


QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE


class Widget : public QWidget
{
    Q_OBJECT


public:
    Widget(QWidget *parent = nullptr);
    ~Widget();


private slots:
    void on_connectBtn_clicked();
    void connected_slot();           //自定义处理connected信号的槽函数
    void readyRead_slot();           //自定义处理readyRead信号的槽函数
    void disconnected_slot();        //自定义处理disconnected信号的槽函数
    void on_sendBtn_clicked();
    void on_disConnectBtn_clicked();


private:
    Ui::Widget *ui;
    //定义通信用的变量
    QTcpSocket *client;         //定义套接字指针
    QString userName;           //用户名


};
#endif // WIDGET_H

widget.cpp

cpp 复制代码
#include "widget.h"
#include "ui_widget.h"


Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    //实例化客户端对象
    client = new QTcpSocket(this);

    //当客户端与服务器建立联系后,如果客户端接受到服务器发来的消息
    //客户端自身就会自动发射一个 readyRead的信号,我们可以将该信号连接到自定义的槽函数中执行相关逻辑
    connect(client, &QTcpSocket::readyRead, this, &Widget::readyRead_slot);

    //如果当前客户端成功连接的服务器,那么该客户端就会自动发射一个connected的信号
    //我们可以将该信号连接到自定义的槽函数中处理相关逻辑
    connect(client, &QTcpSocket::connected, this, &Widget::connected_slot);

    //当客户端断开了与服务器的连接后,该客户端就会自动发射一个disconnected的信号
    //我们可以将该信号连接到自定义的槽函数中处理相关逻辑
    connect(client, &QTcpSocket::disconnected, this, &Widget::disconnected_slot);
}


Widget::~Widget()
{
    delete ui;
}


//连接服务器按钮对应的槽函数
void Widget::on_connectBtn_clicked()
{
    //获取ui界面上的数据
    QString ip = ui->ipEdit->text();     //ip地址
    quint16 port = ui->portEdit->text().toUInt();    //端口号
    userName = ui->nameEdit->text();              //用户名

    //调用套接字成员函数,连接服务器
    //函数原型:void connectToHost(const QHostAddress &address, quint16 port, OpenMode mode = ReadWrite);
    //参数1:要被连接的服务器ip地址
    //参数2:服务器的端口号
    //参数3:默认为可读可写
    //返回值:无
    client->connectToHost(ip, port);
}


//处理connected信号的槽函数的定义
void Widget::connected_slot()
{
    QMessageBox::information(this,"成功","连接成功");
    //将相关组件禁用
    ui->ipEdit->setEnabled(false);
    ui->nameEdit->setEnabled(false);
    ui->portEdit->setEnabled(false);
    ui->connectBtn->setEnabled(false);

    //向服务器发送一条消息
    QString msg = userName + ": 进入聊天室";
    client->write( msg.toLocal8Bit() );
}


//自定义处理readyRead信号的槽函数
void Widget::readyRead_slot()
{
    //从套接字中读取数据
    QByteArray msg = client->readAll();

    //将读取下来的数据展示到ui界面上
    ui->msgListWidget->addItem( QString::fromLocal8Bit(msg) );
}

//消息发送按钮对应的槽函数
void Widget::on_sendBtn_clicked()
{
    //组织要发送的消息
    QString msg = userName + ": " + ui->msgEdit->text();

    //将消息发送给服务器
    client->write(msg.toLocal8Bit());

    //将消息展示到自己界面上
    //准备一个QListWidgetItem类的对象
    QListWidgetItem *item = new QListWidgetItem(msg);
    item->setTextAlignment(Qt::AlignRight);        //将文本右对齐
    ui->msgListWidget->addItem(item);

    //清空消息发送框的内容
    ui->msgEdit->clear();
}

//断开连接按钮对应的槽函数
void Widget::on_disConnectBtn_clicked()
{
    //执行断开连接的操作
    //准备发送消息给服务器
    QString msg = userName + ": 离开聊天室";
    client->write(msg.toLocal8Bit());

    //断开连接
    client->disconnectFromHost();
}


//自定义处理disconnected信号的槽函数的定义
void Widget::disconnected_slot()
{
    QMessageBox::information(this, "提示", "成功断开与服务器的连接");

    //将相关组件启用
    ui->ipEdit->setEnabled(true);
    ui->nameEdit->setEnabled(true);
    ui->portEdit->setEnabled(true);
    ui->connectBtn->setEnabled(true);
}
相关推荐
二十雨辰1 小时前
[linux]docker基础
linux·运维·docker
城南vision1 小时前
计算机网络——TCP篇
网络·tcp/ip·计算机网络
饮浊酒2 小时前
Linux操作系统 ------(3.文本编译器Vim)
linux·vim
lihuhelihu2 小时前
第3章 CentOS系统管理
linux·运维·服务器·计算机网络·ubuntu·centos·云计算
矛取矛求2 小时前
Linux系统性能调优技巧
linux
One_Blanks2 小时前
渗透测试-Linux基础(1)
linux·运维·安全
Perishell3 小时前
无人机避障——大疆与Airsim中的角速度信息订阅获取
linux·动态规划·无人机
爱吃喵的鲤鱼3 小时前
linux进程的状态之环境变量
linux·运维·服务器·开发语言·c++
dessler3 小时前
Linux系统-ubuntu系统安装
linux·运维·云计算
Tony聊跨境3 小时前
独立站SEO类型及优化:来检查这些方面你有没有落下
网络·人工智能·tcp/ip·ip