QT的TcpServer

news2025/1/9 1:15:20

Server服务器端

QT版本5.6.1

  1. 界面设计
    alt text
  2. 工程文件:

添加 network 模块

alt text

  1. 头文件引入TcpServer类和TcpSocketQTcpServer和QTcpSocket
#include <QTcpServer>
#include <QTcpSocket>

alt text

  1. 创建server对象并实例化:
   /*h文件中*/
    QTcpServer *tcpServer;
   /*cpp文件中*/
   tcpServer = new QTcpServer(this);

使用This作为父对象,方便结束进程生命周期时,释放内存空间

  1. 构建槽函数:
    alt text
private slots:
    void newConnection_Slot();
    void readyRead_Slot();
    void stateChanged_Slot(QAbstractSocket::SocketState);

: 在C++的类中声明函数时可以不写参数名只写参数类型,但是,在实现时必须写出参数名。例如 : void stateChanged_Slot(QAbstractSocket::SocketState);形参是枚举类型,但是没有具体的形参名

newConnection_Slot: 当有客户端的socket建立连接时,tcpserver会发送newConnection()信号,触发连接槽函数,用于创建一个新的socket与客户端对接
connect(tcpServer,SIGNAL(newConnection()),this,SLOT(newConnection_Slot()));

void Widget::newConnection_Slot()
{


   ui->recvEdit->append("Have a clinet connection");
   QTcpSocket *temptcpSocket = tcpServer->nextPendingConnection();
   connect(temptcpSocket,SIGNAL(readyRead()),this,SLOT(readyRead_Slot()));
   connect(temptcpSocket,SIGNAL(stateChanged(QAbstractSocket::SocketState)),this,SLOT(stateChanged_Slot(QAbstractSocket::SocketState)));


    // 客户端的ip地址
    QString ipaddr = temptcpSocket->peerAddress().toString();
    quint16 port = temptcpSocket->peerPort();
    // 打印客户端连接的端口信息
    ui->recvEdit->append("服务端:客户端的ip地址:" + ipaddr);
    ui->recvEdit->append("服务端:客户端的端口:" + QString::number(port));

}

readyRead_Slot : 当有客户端的socket发送数据过来,服务端用来与客户端对接的socket发送数据可读信号readyRead(),触发槽函数,读数据
connect(tcpSocket,SIGNAL(readyRead()),this,SLOT(readyRead_Slot()));

void Widget::readyRead_Slot()
{
    //sender()返回触发该函数的信号所属者的指针
    QTcpSocket *tcpSocket = (QTcpSocket *)sender();
    QString buf;

    buf = tcpSocket->readAll();

    ui->recvEdit->append(buf);

}

stateChanged_Slot(QAbstractSocket::SocketState) : 当服务端的socket与客户端断开连接时,会发送连接状态改变信号,触发槽函数

void Widget::stateChanged_Slot(QAbstractSocket::SocketState socketState)
{
    QTcpSocket *tcpSocket = (QTcpSocket *)sender();
    switch(socketState){
    case QAbstractSocket::UnconnectedState:
        ui->recvEdit->append("connection end");
        tcpSocket->deleteLater();
    break;

    default:
        break;
    }

}

  1. ui按键槽函数:
    void on_openBt_clicked();
    void on_colseBt_clicked();
    void on_sendBt_clicked();

void on_openBt_clicked() : 打开服务端

void Widget::on_openBt_clicked()
{
   //判断输入的端口号是否为空,为空则直接返回
      if(ui->portEdit->text().isEmpty()){
          ui->recvEdit->append("port is empty!!");
          return;
      }
      //启动监听,QHostAddress::Any表示所有IP地址
     tcpServer->listen(QHostAddress::Any, ui->portEdit->text().toUInt());
     ui->recvEdit->append("open successful");
     ui->openBt->setEnabled(false);
     ui->colseBt->setEnabled(true);
}

on_colseBt_clicked : 关闭客户端

void Widget::on_colseBt_clicked()
{
   //关闭tcpserver知识关闭的监听功能,不会有新的客户端建立连接,但是已经建立连接的socket仍然可以通信
    tcpServer->close();
    ui->recvEdit->append("close successful");
    ui->openBt->setEnabled(true);
    ui->colseBt->setEnabled(false);
}

on_sendBt_clicked : 发送数据

void Widget::on_sendBt_clicked()
{
    //返回server端与客户端建立连接的所有socket
    QList <QTcpSocket *> socketList = tcpServer->findChildren<QTcpSocket *>();
    qDebug() << "tcpSocket 数量:" << socketList.count() << endl;

    if (socketList.count() == 0) {
        ui->recvEdit->append("当前没有客户端连接,请先与客户端连接!");
        return;
    }
    foreach (QTcpSocket *tmpTcpSocket, socketList) {
        // 服务端向每个客户端发送消息
        tmpTcpSocket->write(ui->sendEdit->text().toUtf8());
    }
    ui->recvEdit->append("服务端:" + ui->sendEdit->text());
}

工程源码:

widget.h:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpServer>
#include <QTcpSocket>


namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
    void state_Slot(QAbstractSocket::SocketState);

private:
    Ui::Widget *ui;
    QTcpServer *tcpServer;

private slots:
    void newConnection_Slot();
    void readyRead_Slot();
    void stateChanged_Slot(QAbstractSocket::SocketState);


    void on_openBt_clicked();
    void on_colseBt_clicked();
    void on_sendBt_clicked();
};

#endif // WIDGET_H


widget.cpp

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

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    tcpServer = new QTcpServer(this);

    connect(tcpServer,SIGNAL(newConnection()),this,SLOT(newConnection_Slot()));
}



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

void Widget::newConnection_Slot()
{


     ui->recvEdit->append("Have a clinet connection");
   QTcpSocket *temptcpSocket = tcpServer->nextPendingConnection();
    connect(temptcpSocket,SIGNAL(readyRead()),this,SLOT(readyRead_Slot()));
    connect(temptcpSocket,SIGNAL(stateChanged(QAbstractSocket::SocketState)),this,SLOT(stateChanged_Slot(QAbstractSocket::SocketState)));


    // 客户端的ip地址
    QString ipaddr = temptcpSocket->peerAddress().toString();
    quint16 port = temptcpSocket->peerPort();
    // 打印客户端连接的端口信息
    ui->recvEdit->append("服务端:客户端的ip地址:" + ipaddr);
    ui->recvEdit->append("服务端:客户端的端口:" + QString::number(port));

}

void Widget::readyRead_Slot()
{
    //sender()返回触发该函数的信号所属者的指针
    QTcpSocket *tcpSocket = (QTcpSocket *)sender();
    QString buf;

    buf = tcpSocket->readAll();

    ui->recvEdit->append(buf);

}

void Widget::stateChanged_Slot(QAbstractSocket::SocketState socketState)
{
    QTcpSocket *tcpSocket = (QTcpSocket *)sender();
    switch(socketState){
    case QAbstractSocket::UnconnectedState:
        ui->recvEdit->append("connection end");
        tcpSocket->deleteLater();
    break;
    default:
        break;
    }

}



void Widget::on_openBt_clicked()
{
      if(ui->portEdit->text().isEmpty()){
          ui->recvEdit->append("port is empty!!");
          return;
      }
     tcpServer->listen(QHostAddress::Any, ui->portEdit->text().toUInt());
     ui->recvEdit->append("open successful");
     ui->openBt->setEnabled(false);
     ui->colseBt->setEnabled(true);
}

void Widget::on_colseBt_clicked()
{
    tcpServer->close();
    ui->recvEdit->append("close successful");
    ui->openBt->setEnabled(true);
    ui->colseBt->setEnabled(false);
}

void Widget::on_sendBt_clicked()
{
    //返回server端与客户端建立连接的所有socket
    QList <QTcpSocket *> socketList = tcpServer->findChildren<QTcpSocket *>();
    qDebug() << "tcpSocket 数量:" << socketList.count() << endl;

    if (socketList.count() == 0) {
        ui->recvEdit->append("当前没有客户端连接,请先与客户端连接!");
        return;
    }
    foreach (QTcpSocket *tmpTcpSocket, socketList) {
        // 服务端向每个客户端发送消息
        tmpTcpSocket->write(ui->sendEdit->text().toUtf8());
    }
    ui->recvEdit->append("服务端:" + ui->sendEdit->text());
}

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

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

相关文章

Python根据预设txt生成“你画我猜”题目PPT(素拓活动小工具)

Python根据预设txt生成“你画我猜”题目PPT&#xff08;素拓活动小工具&#xff09; 场景来源 去年单位内部的一次素拓活动&#xff0c;分工负责策划设置其中的“你画我猜”环节&#xff0c;网络上搜集到题目文字后&#xff0c;想着如何快速做成对应一页一页的PPT。第一时间想…

观察者模式实战:解密最热门的设计模式之一

文章目录 前言一、什么是观察者模式二、Java实现观察者模式2.1 观察者接口2.2 具体观察者2.3 基础发布者2.4 具体发布者2.5 消息发送 三、Spring实现观察者模式3.1 定义事件类3.2 具体观察者3.3 具体发布者3.4 消息发送 总结 前言 随着系统的复杂度变高&#xff0c;我们就会采…

Google 发布 CodeGemma 7B,8K上下文,性能超CodeLlama 13B

CodeGemma简介 CodeGemma模型是谷歌的社区开放编程模型&#xff0c;专门针对代码领域进行优化。一系列功能强大的轻量级模型&#xff0c;能够执行多种编程任务&#xff0c;如中间代码填充、代码生成、自然语言理解、数学推理和指令遵循。CodeGemma模型是在大约500B个主要为英语…

【算法入门教育赛1D】环形密码 - 字符串 | C++题解与代码

题目链接&#xff1a;https://www.starrycoding.com/problem/161 题目描述 小 e e e有一个宝箱&#xff0c;这个宝箱有一个长度为 n n n的密码&#xff0c;但是这个密码校验器是一个环形&#xff0c;意思是只要密码从任意一位开始读&#xff08;读到最后一位回到第一位继续&a…

每日OJ题_贪心算法二⑤_力扣870. 优势洗牌(田忌赛马)

目录 力扣870. 优势洗牌&#xff08;田忌赛马&#xff09; 解析代码 力扣870. 优势洗牌&#xff08;田忌赛马&#xff09; 870. 优势洗牌 难度 中等 给定两个长度相等的数组 nums1 和 nums2&#xff0c;nums1 相对于 nums2 的优势可以用满足 nums1[i] > nums2[i] 的索引…

Redis - Zset 有序集合

前言 它保留了集合不能有重复成员的特点&#xff0c;但与集合不同的是&#xff0c;有序集合中的每个元素都有⼀个唯⼀的浮点类型的分数&#xff08;score&#xff09;与之关联&#xff0c;有序集合中的元素是可以维护有序性的&#xff0c;但这个有序不是⽤下标作为排序依据⽽是…

笔记13-OSError: [Errno 24] Too many open files

文章目录 参考文献失败尝试系列查看发现&#xff0c;似乎是因为线程数有限制 修改配置先查查看 增加文件数限制&#xff0c;然后使用命令运行&#xff08;成功&#xff09; 参考文献 Linux 最大可以打开多少文件描述符&#xff1f; OSError: [Errno 24] Too many open files错…

Redis-单机安装

试图从官网注册不了我也不知道什么情况。 网盘自取吧&#xff0c;链接&#xff1a;https://pan.baidu.com/s/1KERBQaH9gCT10AGt9z0_jg?pwdyjen 安装比较简单&#xff0c;照着敲就完了每一步都试过了&#xff0c;先单机安装&#xff0c;后面搭建集群。 1.将安装包放到/usr/…

一文带你了解MySQL的索引分类

文章目录 ☃️分类☃️演示图☃️思考☃️总结 欢迎来到 请回答1024 的博客 &#x1f353;&#x1f353;&#x1f353;欢迎来到 请回答1024的博客 关于博主&#xff1a; 我是 请回答1024&#xff0c;一个追求数学与计算的边界、时间与空间的平衡&#xff0c;0与1的延伸的后端开…

C++之set/map相关实现

看着上面的图片&#xff0c;你可能对set和map的多样变化产生疑惑&#xff0c;下面我们就来详细讲解他们的区别以及实现 一.set/map 首先&#xff0c;在这里我们要声明&#xff0c;如果你对二叉搜索树一点都不了解的话&#xff0c;建议你先去将搜索二叉树学会再来学习这里的内…

MFC 列表控件删除实例(源码下载)

1、本程序基于前期我的博客文章《MFC下拉菜单打钩图标存取实例&#xff08;源码下载) 》 2、程序功能选中列表控件某一项&#xff0c;删除按钮由禁止变为可用&#xff0c;点击删除按钮&#xff0c;选中的项将删除。 3、首先在主界面添加一个删除参数按钮。 4、在myDlg.cpp 文件…

Python语言零基础入门——文件

目录 一、文件的基本概念 1.文件 2.绝对路径与相对路径 3.打开文件的模式 二、文件的读取 三、文件的追加 四、文件的写入 五、with语句 六、csv文件 1.csv文件的读取 2.csv文件的写入 七、练习题&#xff1a;实现日记本 一、文件的基本概念 1.文件 文件是以计算…

win10禁止自动更新的终极方法

添加注册表值 1.运行&#xff0c;输入regedit 2.打开注册表编辑器依次进入以下路径“计算机\HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings”。 3.在Settings项中&#xff0c;新建DWORD&#xff08;32位&#xff09;值(D)&#xff0c;重命名为以下命名“Fl…

python判断大图中包含小图并输出位置总结

python判断大图中包含小图并输出位置总结 没啥可说的&#xff0c;项目遇到了就直接上代码&#xff0c;可以减轻劳动力&#xff0c;花最少得时间实现应用功能。 import cv2 # 读取大图片和小图片的路径 img_big cv2.imread(big_image.png) img_small cv2.imread(small_image…

使用protoc-jar-maven-plugin生成grpc项目

在《使用protobuf-maven-plugin生成grpc项目》中我们使用protobuf-maven-plugin完成了grpc代码的翻译。本文我们将只是替换pom.xml中的部分内容&#xff0c;使用protoc-jar-maven-plugin来完成相同的功能。总体来说protoc-jar-maven-plugin方案更加简便。 环境 见《使用proto…

数据结构--顺序表经典OJ题

例1&#xff1a;合并有序顺序表 给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2&#xff0c;另有两个整数 m 和 n &#xff0c;分别表示 nums1 和 nums2 中的元素数目。请你 合并 nums2 到 nums1 中&#xff0c;使合并后的数组同样按 非递减顺序 排列。 注意&#xff…

R可视化:分组频率分布直方图和密度图

介绍 ggplot2绘制分组频率分布直方图和密度图 加载R包 knitr::opts_chunk$set(message FALSE, warning FALSE) library(tidyverse) library(patchwork) library(ggpubr) library(rstatix)# rm(list ls()) options(stringsAsFactors F) options(future.globals.maxSize …

数据结构与算法---树

数据结构可视化网址 Structure Visualization: https://www.cs.usfca.edu/~galles/visualization/Totuma: https://www.totuma.cn/Algorithm Visualizer: https://algorithm-visualizer.org/ 构建二叉树 // C#include<stdio.h> #include<stdlib.h>typedef char T…

电脑找不到msvcp140.dll如何修复?msvcp140.dll丢失的多种解决方法分享

在日常电脑操作过程中&#xff0c;用户可能会遇到一个令人困扰的问题&#xff0c;即屏幕上突然弹出一条错误提示&#xff1a;“由于找不到msvcp140.dll&#xff0c;无法继续执行代码”。这一情况往往导致应用程序无法正常启动或运行&#xff0c;给工作和娱乐带来不便。不过&…

freertos入门---创建FreeRTOS工程

freertos入门—创建FreeRTOS工程 1 STM32CubeMx配置 双击运行STM32CubeMX,在首页选择“ACCESS TO MCU SELECTOR”,如下图所示&#xff1a;   在MCU选型界面&#xff0c;输入自己想要开发的芯片型号&#xff0c;如&#xff1a;STM32F103C8T6: 2 配置时钟 在“System Core”…