QT 与 C++实现基于[ TCP ]的聊天室界面

news2025/1/23 3:56:54

TCP客户端 

Widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpSocket>   //客户端类
#include <QMessageBox>
#include <QListWidgetItem>
#include <QDebug>


QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

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

public slots:
    void connected_slot();//connected信号对应槽函数的声明
    void readyRead_slot();
    void disconnected_slot();

private slots:
    void on_connectbtn_clicked();

    void on_sendbtn_clicked();

    void on_disconnectbtn_clicked();

private:
    Ui::Widget *ui;
    QString msgfor="";

    //实例化一个客户端指针
    QTcpSocket *socket;

    //定义一个变量存储用户名
    QString userName;
};
#endif // WIDGET_H

Widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <string>
#include<iostream>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
    ,socket(new QTcpSocket(this))//给客户端实例化空间
{
    ui->setupUi(this);

    //初始化界面,设置为不可用
    ui->msgEdit->setEnabled(false);
    ui->sendbtn->setEnabled(false);
    ui->disconnectbtn->setEnabled(false);

    //如果成功连接服务器,那么客户端就会自动发射一个connected()信号
    //将该信号连接到自定义的槽函数,书写逻辑代码。由于只需要连接一次,所以连接函数写在构造函数中
    connect(socket, &QTcpSocket::connected, this, &Widget::connected_slot);

    //此时说明客户端和服务器已经成功建立连接,如果服务器发来数据,那么客户端就会自动发射readyRead()信号
    //将该信号连接到自定义的槽函数中,读取数据。由于只需要连接一次,所以连接函数写在构造函数中
    connect(socket, &QTcpSocket::readyRead, this, &Widget::readyRead_slot);

    //如果成功与服务器断开连接,那么客户端就会自动发射disconnected信号
    //将信号连接到自定义的槽函数中处理逻辑代码 ,由于只需连接一次,所以连接函数写在构造函数中
    connect(socket, &QTcpSocket::disconnected, this, &Widget::disconnected_slot);
}

Widget::~Widget()
{
    delete ui;
}
//connected信号对槽函数实现
void Widget::connected_slot()
{
    //连接成功
    QMessageBox::information(this,"","连接服务器成功");

    //告诉服务器 我来了
    userName = ui->userNameEdit->text();
    //组织语言
    QString msg = userName + ": 进入聊天室";
    //将信息发送给服务器
    socket->write(msg.toLocal8Bit());

    //此时说明客户端和服务器已经成功建立连接,如果服务器发来数据,那么客户端就会自动发射readyRead()信号
    //将该信号连接到自定义的槽函数中,读取数据。由于只需要连接一次,所以连接函数写在构造函数中

    ui->msgEdit->setEnabled(true); //设置可用
    ui->sendbtn->setEnabled(true);
    ui->disconnectbtn->setEnabled(true);

    ui->userNameEdit->setEnabled(false);//设置不可用
    ui->ipEdit->setEnabled(false);
    ui->portEdit->setEnabled(false);
    ui->connectbtn->setEnabled(false);
}

//readyRead()信号对应槽函实现
void Widget::readyRead_slot()
{
    //读取服务器发来的数据
    QByteArray msg = socket->readAll();

    //将信息数据放入ui界面上
    QListWidgetItem *aItem;
    aItem=new QListWidgetItem();

    int num=userName.size();
    QString mmsg=QString::fromLocal8Bit(msg);

    //截取用户名
    QByteArray bytesSub = msg.left(num); // 截取字节
    QString mmsg1=QString::fromLocal8Bit(bytesSub);

    //判断用户名是否是自己
    if(mmsg1!=userName)
    {
        aItem->setText(mmsg);
        ui->listWidget->addItem(aItem);
    }
    else
    {
        QString msg2=msgfor+":"+userName;
        aItem->setText(msg2);
        aItem->setTextAlignment(Qt::AlignRight);
        ui->listWidget->addItem(aItem);
    }
}

//disconnected信号对应的槽函数实现
void Widget::disconnected_slot()
{
    ui->msgEdit->setEnabled(false); //设置不可用
    ui->sendbtn->setEnabled(false);
    ui->disconnectbtn->setEnabled(false);

    ui->userNameEdit->setEnabled(true);//设置可用
    ui->ipEdit->setEnabled(true);
    ui->portEdit->setEnabled(true);
    ui->connectbtn->setEnabled(true);
}

//---连接服务器按钮对应的槽函数
void Widget::on_connectbtn_clicked()
{
    //获取ui界面上的ip 和 端口号
    QString ip = ui->ipEdit->text();
    quint16 port = ui->portEdit->text().toUInt(); //将字符串转换整型

    //让客户端连接服务器
    //函数原型:virtual void connectToHost(const QString &hostName, quint16 port, OpenMode mode = ReadWrite, NetworkLayerProtocol protocol = AnyIPProtocol);
    //参数一:服务器的ip地址
    //参数二:服务器的端口号
    socket->connectToHost(ip, port);

    //如果成功连接到服务器,客户端就会自动发射一个connected信号进行连接
    //我们就可以将该信号连接到自定义槽函数中处理逻辑代码,由于只需要连接一次
    //所以将连接函数写在构造函数中
}

//发送按钮对应的槽函数处理
void Widget::on_sendbtn_clicked()
{
    //获取Ui界面上的内容
    QString msg = ui->msgEdit->text();
    msgfor=msg;

    msg = userName + ": " + msg;

//    //自己的消息放右边显示
//    QListWidgetItem *aItem=new QListWidgetItem();
//    QString msg2=msgfor+":"+userName;
//    aItem->setText(msg2);
//    aItem->setTextAlignment(Qt::AlignRight);
//    ui->listWidget->addItem(aItem);

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

    //清空行编辑器
    ui->msgEdit->clear();
}

//断开连接按钮 对应的槽函数
void Widget::on_disconnectbtn_clicked()
{
    //告诉服务 我走了
    QString msg = userName + ":  优雅的离开了聊天室";

    socket->write(msg.toLocal8Bit());

    //将客户端与服务器断开连接
    socket->disconnectFromHost();

    //如果成功与服务器断开连接,那么客户端就会自动发射disconnected信号
    //将信号连接到自定义的槽函数中处理逻辑代码 ,由于只需连接一次,所以连接函数写在构造函数中
}

TCP服务器

Widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpServer>   //服务器类
#include <QMessageBox>  //消息对话框类
#include <QDebug>
#include <QTcpSocket>   //客户端类
#include <QList>        //链表容器

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

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

public slots:
    void newConnect_slots();
    void readyRead_slot();

private slots:
    void on_startbtn_clicked();

private:
    Ui::Widget *ui;

    //实例化一个服务器指针
    QTcpServer *server;

    //定义一个存放客户端的容器
    QList<QTcpSocket*> socketList;

};
#endif // WIDGET_H

Widget.cpp

#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
    ,server(new QTcpServer(this))//给服务器对象实例化具体的空间
{
    ui->setupUi(this);
}

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

//newConnect信号对应的槽函数
void Widget::newConnect_slots()
{
    //有新的用户连接
    qDebug() << "有新的用户连接...";

    //获取最新连接的客户端套接字
    //函数原型:virtual QTcpSocket *nextPendingConnection();
    QTcpSocket *s=server->nextPendingConnection();

    //将获取的客户端放入客户端容器中
    socketList.push_back(s);

    //程序运行至此,此时说明服务器和客户端已经建立了连接
    //如果有客户端向服务器发来数据,客户端就会自动发射一个readyRead信号
    //就可以将该信号连接到自定义的槽函数中,读取数据
    connect(s, &QTcpSocket::readyRead, this, &Widget::readyRead_slot);
}

//readyRead信号对应的槽函数
void Widget::readyRead_slot()
{
    //遍历客户端容器,移除无效客户端
    for(int i=0; i<socketList.count(); i++)
    {
        //判断客户端和服务器的连接状态
        //函数原型:SocketState state() const;
        //函数返回值 枚举值为0的表示未连接的
        if( socketList.at(i)->state() == 0)
        {
            //删除该元素
            socketList.removeAt(i);
        }
    }

    //遍历客户端容器,寻找哪个客户端有数据待读
    for(int i=0; i<socketList.count(); i++)
    {
        //函数功能:数据的字节
        //函数原型:qint64 bytesAvailable() const override;
        if( socketList.at(i)->bytesAvailable() != 0) //说明有数据
        {
            //读取客户端发来的数据
            QByteArray msg = socketList.at(i)->readAll();

            //将读取到的数据 放入ui界面上
            ui->listWidget->addItem(QString::fromLocal8Bit(msg));

            //将数据广播给所有客户端
            for(int j=0; j<socketList.count(); j++)
            {
                //if(j!=i),第二种方法
                socketList.at(j)->write(msg);
            }
        }
    }
}

//启动服务器按钮对应的槽函数
void Widget::on_startbtn_clicked()
{
    //获取ui界面的端口号
    //将字符串转换成整形
    quint16 port=ui->portlineEdit->text().toUInt();

    //服务器设置监听
    //函数原型: bool listen(const QHostAddress &address = QHostAddress::Any, quint16 port = 0);
    //参数1:监听的主机,可以是指定主机,也可以任意
    //参数2:监听的端口号,可以是指定,也可以系统提供
    //返回值:监听成功返回true  否则false
    if(server->listen(QHostAddress::Any,port))
    {
        //监听成功
        QMessageBox::information(this,"","启动服务器成功!");
    }
    else
    {
        //监听失败
        QMessageBox::information(this,"","启动服务器失败!");
        return;
    }

    //此时服务器已经设置好监听,如果有客户端发来连接,那么服务器端就会自动发射一个newConnection()信号
    //将该信号连接到自定义的槽函数中,处理逻辑代码
    connect(server, &QTcpServer::newConnection, this, &Widget::newConnect_slots);
}

 思维导图

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2062075.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于YOLO V8的PCB缺陷检测识别系统(python源码+Pyqt5界面+数据集+训练代码)

数据集准备&#xff1a;收集并标注PCB缺陷的图像。模型训练&#xff1a;使用YOLO v8框架训练一个模型来识别这些缺陷。GUI开发&#xff1a;利用PyQt5创建一个用户友好的图形界面。模型部署&#xff1a;在GUI中集成训练好的模型&#xff0c;使用户能够上传PCB图像并得到缺陷检测…

数据结构-树(二叉树)-第三天

结合leetcode学习c 学习资料来源 定义 二叉树&#xff08;binary tree&#xff09;是一种非线性数据结构&#xff0c;代表“祖先”与“后代”之间的派生关系&#xff0c;体现了“一分为二”的分治逻辑。 与链表类似&#xff0c;二叉树的基本单元是节点&#xff0c;每个节点包含…

[数据集][目标检测]停车场空位检测数据集VOC+YOLO格式7959张2类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;7959 标注数量(xml文件个数)&#xff1a;7959 标注数量(txt文件个数)&#xff1a;7959 标注…

p9 镜像的基本命令

docker的常用命令 帮助命令 查看docker的版本 docker version2. 查看当前安装的docker的详细的信息 docker的系统信息包括镜像和容器的数量 docker info3. 查询帮助命令 image是你想要获取哪一个命令的帮助文档,一般百度或者人工智能就行所有命令的可选项都可以用这个–hel…

DN专业3D图形制作软件win/mac软件安装下载(附下载链接)

目录 一、软件概述 1.1 Adobe DN简介 1.2 Windows/Mac系统要求 Windows系统&#xff1a; Mac系统&#xff1a; 二、安装步骤 2.1 下载与解压 2.2 安装程序 2.3 启动软件 三、使用教程 3.1 界面介绍 3.2 创建和编辑3D内容 3.3 合成与渲染 四、高级技巧与注意事项 …

.NET内网实战:通过DNS查找出网主机

01阅读须知 此文所节选自小报童《.NET 内网实战攻防》专栏&#xff0c;主要内容有.NET在各个内网渗透阶段与Windows系统交互的方式和技巧&#xff0c;对内网和后渗透感兴趣的朋友们可以订阅该电子报刊&#xff0c;解锁更多的报刊内容。 02基本介绍 03编码实现 在红队活动中&…

Modbus 数据帧解析

Modbus 是一种通信协议&#xff0c;广泛应用于工业自动化系统中&#xff0c;用于连接电子设备。它是一种基于主从结构的通信协议&#xff0c;其中主设备可以与多个从设备通信。数据通过一系列的帧进行传输&#xff0c;每一帧包含多个字段&#xff0c;每个字段都有特定的功能。 …

SQL进阶技巧:最近有效的缺失值填充问题【last_value实现版】

目录 0 场景描述 1 数据准备 2 问题分析 3 小结 0 场景描述 场景:现在有一张商品入库表,包括商品id、商品成本和入库日期3个字段,由于某些原因,导致部分商品的成本缺失(为0或者没有值都是缺失),这样不利于我们计算成本。所以现在要把缺失的商品进价补充完整,补充的…

OpenCTI:开源网络威胁情报平台

OpenCTI 是一个开源平台&#xff0c;旨在帮助组织管理其网络威胁情报 (CTI) 数据和可观察数据。 该平台由 Filigran 开发&#xff0c;使用基于 STIX2 标准的知识模式构建数据。 它采用现代 Web 应用程序架构&#xff0c;配备 GraphQL API 和用户友好的前端。 OpenCTI 与 MIS…

MySQL在Windows和Ubuntu上的安装与远程连接配置

MySQL是一个广泛使用的开源关系数据库管理系统&#xff0c;适用于各种操作系统。本文将详细介绍如何在Windows和Ubuntu系统上安装MySQL&#xff0c;并配置远程连接。 1. 在Windows上安装MySQL 1.1 下载MySQL安装包 首先&#xff0c;访问MySQL官方网站&#xff08;https://de…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《考虑模块冗余和系统状态的多...电系统可靠性的蒙特卡洛分析》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

【Java日志系列】JCL、SLF4J日志门面

目录 前言 一、日志门面概述 1. 门面模式&#xff08;外观模式&#xff09; 2. 日志门面 二、JCL 1. JCL组件结构 2. JCL案例 三、SLF4J 1. SLF4J简介 2. SLF4J桥接技术 3. 快速入门 4. SLF4J集成日志实现 4.1 集成nop 4.2 集成Logback 4.3 集成Log4j 4.4 集成…

一文彻底搞懂Transformer - 总体架构,零基础入门到精通,收藏这一篇就够了

Transformer 一、RNN编码器-解码器架构**** ********序列到序列模型&#xff08;Seq2Seq&#xff09;&#xff1a;Seq2Seq模型的目标是将一个输入序列转换成另一个输出序列&#xff0c;这在多种应用中都具有广泛的实用价值&#xff0c;例如语言建模、机器翻译、对话生成等。 …

网络及ipc内存共享

大字符串找小字符串 调试 1. 信号处理函数注册&#xff1a;•一旦使用 signal 函数注册了信号处理函数&#xff0c;该函数就会一直有效&#xff0c;直到程序结束或者显式地取消注册。2. 注册多次的影响&#xff1a;•如果多次注册同一信号的处理函数&#xff0c;最后一次注册的…

Java 入门指南:List 接口

Collection 接口提供了一系列用于操作和管理集合的方法&#xff0c;包括添加、删除、查询、遍历等。它是所有集合类的根接口&#xff0c;包括 List、Set、Queue 等。 Collection 接口常见方法 add(E element)&#xff1a;向集合中添加元素。 addAll(Collection col)&#xff1…

打架检测算法在公共安全中的创新应用

在现代社会中&#xff0c;公共场所的安全问题日益受到重视&#xff0c;特别是暴力事件的监控和预防更是各类安防系统的核心需求之一。随着视频监控设备的普及&#xff0c;如何从大量监控数据中实时、高效地识别出潜在的暴力行为&#xff0c;成为亟待解决的难题。传统的视频监控…

怎么对前端的一些按钮做一个权限校验

在一般情况下,我们需要对一些按钮做一个权限校验,来保证只有有权限的用户才能看到 1.创建一个js文件,来写我们的全局方法 我的方法是这样的 import Vue from vue;Vue.mixin({methods:{hasAuth(perm) {var authority this.$store.state.menu.permList;if (authority.indexOf(…

JointJs 在 Vue 中的使用探索(一): Hello JointJS

文章目录 前言根本问题探索过程安装 jointjs测试 joint/core demo 代码jointjs joint/core 前言 最近需要用到 JointJs 做一些东西&#xff0c;但是 官方文档 的 joint/core 跑下来后发现并不太好使&#xff0c;空白一片…&#xff08;这是个误会…&#xff09; 所以开了个贴给…

map与set容器初识:初步运用map与set

前言&#xff1a; 本文主要讲解的时对于map与set容器的初步使用&#xff0c;希望大家对map与set容器不熟悉的看了之后可以快速运用set与map到日常中来。&#xff08;本文适合对vector等基础容器有一定基础的同学&#xff09; 一、set与map容器常见接口 迭代器接口与以往的所…

【hot100篇-python刷题记录】【不同路径】

R5-多维动态规划篇 多维动态规划的核心在于建立多维状态记录表。 本题中&#xff0c;建立dp二维数组表&#xff08;初始化为1&#xff09; dp[i][j]dp[i-1][j]dp[i][j-1] 注意&#xff0c;需要判断是否存在&#xff0c;因为二维数组有边界 第一种处理需要判断边界 第二种&…