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

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_NAMESPACEclass Widget : public QWidget
{Q_OBJECTpublic: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_NAMESPACEclass Widget : public QWidget
{Q_OBJECTpublic: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  否则falseif(server->listen(QHostAddress::Any,port)){//监听成功QMessageBox::information(this,"","启动服务器成功!");}else{//监听失败QMessageBox::information(this,"","启动服务器失败!");return;}//此时服务器已经设置好监听,如果有客户端发来连接,那么服务器端就会自动发射一个newConnection()信号//将该信号连接到自定义的槽函数中,处理逻辑代码connect(server, &QTcpServer::newConnection, this, &Widget::newConnect_slots);
}

 思维导图

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

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

相关文章

【操作系统】实验:文件系统

目录 一、实验目的 二、实验要求 三、实验步骤 四、核心代码 五、记录与处理 六、思考 七、完整报告和成果文件提取链接 一、实验目的 1、掌握文件系统的基本结构和文件系统的管理方法 2、加深对两级文件目录认识和理解 3、对文件操作的系统命令实质内容和执行过程深入…

【9月持续更新】国内ChatGPT-4o中文镜像网站整理~

以前我也是通过官网使用&#xff0c;但是经常被封号&#xff0c;就非常不方便&#xff0c;后来有朋友推荐国内工具&#xff0c;用了一阵之后&#xff0c;发现&#xff1a;稳定方便&#xff0c;用着也挺好的。 最新的 GPT-4o、4o mini&#xff0c;可搭配使用~ 1、 最新模型科普&…

Revit 2025.2新增功能及安装教程和下载

Revit 2025.2 带来了许多新功能和改进&#xff0c;增强了BIM工作流程。以下是主要的新功能&#xff1a; 项目浏览器标签功能&#xff1a;新增的标签功能使得用户可以通过标签过滤浏览器列表&#xff0c;这对大型项目非常有用&#xff0c;减少了滚动和点击次数​。 链接管理对…

若依将登录用户的userId自动加载到查询中

点击搜索&#xff0c;会将登录用户的userId作为搜索条件&#xff0c;去查询。 新版本自动存储了userId&#xff0c;我们不用改&#xff0c;只要知道如何引用。 前端使用 在对应的vue文件&#xff0c;查询queryParams 加查询的值 然后参考他的 添加store import store from &…

PDM系统多少钱一般?一文详细了解三品PDM系统报价

在现代企业的数字化转型过程中&#xff0c;PDM产品数据管理软件扮演着至关重要的角色。PDM软件是一种应用于管理产品全生命周期的软件工具&#xff0c;涵盖了产品设计、工艺流程、生产制造、销售和维护等多个环节。随着技术的不断进步和市场的日益竞争&#xff0c;PDM软件的价格…

零基础学PLC的指令-沿指令(2)

扫描操作数的信号上升沿&#xff1a; 在触点分配的 "IN" 位上检测到正跳变&#xff08;0->1&#xff09;时&#xff0c;该触点的状态为 TRUE。该触点逻辑状态随后与能流输入状态组合以设置能流输出状态。P 触点可以放置在程序段中除分支结尾外的任何位置。 扫描…

AOP思想

什么是AOP AOP&#xff1a;Aspect oriented programming 面向切面编程&#xff0c;AOP 是 OOP&#xff08;面向对象编程&#xff09;的一种延续。 解决&#xff1a;在不改变原有业务逻辑的情况下&#xff0c;增强横切逻辑代码&#xff0c;根本上解耦合&#xff0c;避免横切逻…

大规模预训练语言模型的参数高效微调

人工智能咨询培训老师叶梓 转载标明出处 大规模预训练语言模型&#xff08;PLMs&#xff09;在特定下游任务上的微调和存储成本极高&#xff0c;这限制了它们在实际应用中的可行性。为了解决这一问题&#xff0c;来自清华大学和北京人工智能研究院的研究团队探索了一种优化模型…

四川凭什么能成为中国的战略腹地?

四川因其独特的地理位置、丰富的资源、雄厚的工业基础、庞大的人口及复杂多样的地形等&#xff0c;就在今年1月份的时候&#xff0c;国务院正式批复了一份重磅文件&#xff0c;四川被国务院正式定位为战略腹地&#xff0c;即全体中国人的退路和国家的备份省。 那么四川凭什么能…

STM32(F103ZET6)第四课:串口中断

目录 需求一、串口中断过程与作用二、中断实现流程1.中断优先级分组2.配置串口中断 三、需求的实现 需求 1.设备上电后四个灯灭。 2.按下KEY1&#xff0c;LED1灯亮&#xff0c;同时串口发送“LED1灯亮”。 3.再次按下KEY1&#xff0c;LED1灯灭&#xff0c;同时串口发送“LED1灯…

android studio 新建java工程, 安卓新建项目,android studio2024 如何新建java项目

主要解决&#xff0c;新增安卓工程&#xff0c;没有java选项 1. 点击左上角FIle -> New -> 2. 选择 no activity 选项&#xff0c; 然后next 3. langua 就可以选择java 了。name自己定义项目名称&#xff0c;项目存储地址&#xff0c;包名。 配置完成选择finish. 4. fin…

Rust Web框架怎么选?

在最新的web框架基准测试中 https://www.techempower.com/benchmarkshttps://web-frameworks-benchmark.netlify.app/result?lrust 除去一些没有发布分支或者已经很少维护或者不是rust写的框架外&#xff0c;可以看到Axum、Actix-web是目前性能最好的两款后端框架&#xff0…

python,json数据格式,pyecharts模块,pycharm中安装pyecharts

json数据格式 JSON是一种轻量级的数据交互格式 可以按照JSON指定的格式去组织和封装数据 JSON本质上是一个带有特定格式的字符串 主要功能&#xff1a; json就是一种在各个编程语言中流通的数据格式&#xff0c;负责不同编程语言中的数据传递和交互. 类似于&#xff1a; 国…

手撕python之基本数据类型以及变量

​​​​​​1.基础概念 python就是将不同的数据划分成了不同的类型 就像我们生活中的数据有数字、字符等数据一样 小知识点&#xff1a; 注释&#xff1a;# 全体注释&#xff1a;AltF3 取消注释&#xff1a;AltF4 2.数值类型 数值类型概括 数值类型分为三种&#xff…

JavaWeb JavaScript ⑤ JS常见对象

某一瞬间&#xff0c;是平静的自我接纳 —— 24.8.27 一、数组 1.创建数组的四种方式 ① new Array() 创建空数组 ② new Array(5) 创建数组时给定长度 ③ new Array(ele1,ele2,…,elen); 创建数组时给定元素值 ④ [ele1,ele2,…,elen] 相当于…

【STM32单片机_(HAL库)】3-4-3【中断EXTI】【智能排队控制系统】排队系统代码框架搭建

3-4-2系统框图及硬件接线 3.软件 beep、exti、gate、LCD1602、led、tasks驱动文件添加GPIO常用函数中断配置流程main.c程序 #include "sys.h" #include "delay.h" #include "led.h" #include "tasks.h" #include "gate.h"…

腾讯浙大提出定制化视频生成框架CustomCrafter,只需通过少量图像就可以完成高质量视频生成!

腾讯联合浙大提出了一种定制化视频生成框架-CustomCrafter&#xff0c;它能够基于文本提示和参考图像生成自定义视频&#xff0c;同时保留运动生成和概念组合的能力。通过设计一系列灵活的模块&#xff0c;使得模型实现了无需额外视频&#xff0c;通过少量图像学习&#xff0c;…

嵌入式:用J-Link Commander和J-Flash进行Flash编程的区别

相关阅读 嵌入式https://blog.csdn.net/weixin_45791458/category_12768532.html?spm1001.2014.3001.5482 J-Link Commander和J-Flash都是用于Flash编程的工具&#xff0c;但它们的功能和应用场景有所不同。以下是两者的区别&#xff1a; J-Link Commander: 类型: 命令行工…

机械学习—零基础学习日志(如何理解概率论4)

当已知一个概率&#xff0c;求解另外一个函数的概率。以下是离散型的概率计算方法。 这里是连续型的&#xff0c;已知概念密度&#xff0c;计算对应的另外一个函数的概率。 这里需要求解对应的原始函数。 这里我们做一道练习题。 《概率论与数理统计期末不挂科|考研零基础入门…

什么是Redis大key问题?如何解决?

目录 Key多大算大呢&#xff1f; 识别big key 处理big key Big Key是Redis中存储了大量的数据的Key&#xff0c;不要误以为big key只是表示Key的值很大&#xff0c;他还包括这个Key对应的value占用空间很多的情况&#xff0c;通常在String、list、hash、set、zset等类型中出…