【QT5】<总览五> QT多线程、TCP/UDP

文章目录

前言

一、QThread多线程

二、QT中的TCP编程

1. TCP简介

2. 服务端程序编写

3. 客户端程序编写

4. 服务端与客户端测试

三、QT中的UDP编程

1. UDP简介

2. UDP单播与广播程序


前言

承接【QT5】<总览四> QT常见绘图、图表及动画。若存在版权问题,请联系作者删除!


一、QThread多线程

1. 作用:创建多线程,防止应用程序界面卡顿。

2. 主要操作:

  • ①创建的类需要继承QThread类。
  • ②重写run函数,新建的线程会执行run函数。
  • ③线程开启:对象调用start方法,使线程执行run函数。
  • ④线程终止:对象调用terminate方法,使线程不再执行run函数。
  • ⑤线程销毁:动态申请new需要调用deleteLater方法销毁线程对象。(该函数可以放置于run函数内,当run函数执行完毕后就会销毁该线程对象,防止内存泄漏)注意:尽量少用静态申请栈空间的方式创建线程对象,因为很可能该对象销毁时线程仍在执行,就会报错。

3. 实例演示:创建两个按钮,一个按钮点击时创建新线程,另一个按钮点击时终止新线程。

【1】widget.h:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QThread>

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE


//自定义的线程类
class MyThread : public QThread
{
    Q_OBJECT

public:
    MyThread(QWidget *parent = nullptr){
        Q_UNUSED(parent)//防止编译器警告
    }

    ~MyThread(){
        qDebug("线程销毁");
    }

    void run() override{
        qDebug("线程开始");
        sleep(5);//QThread里才有该方法
        qDebug("线程停止");
        deleteLater();//销毁线程
    }
};


//Widget类
class Widget : public QWidget
{
    Q_OBJECT

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

private slots:
    void on_pushButton_clicked();

    void on_pushButton_2_clicked();

private:
    Ui::Widget *ui;
    MyThread *mythread;
};
#endif // WIDGET_H

【2】widget.cpp:

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

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

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

//“线程开始”按钮的槽函数
void Widget::on_pushButton_clicked()
{
    mythread = new MyThread;
    mythread->start();
}

//“线程终止”按钮的槽函数
void Widget::on_pushButton_2_clicked()
{
    //若线程没有完成,则终止
    if (!mythread->isFinished())
        mythread->terminate();
}

【3】运行效果:

  • ①当我们点击“线程开始”按钮后,控制台打印“线程开始”,五秒后控制台打印“线程停止”和“线程销毁”。
  • ②当我们点击“线程开始”按钮后,控制台打印“线程开始”,当我们在五秒内点击“线程结束”按钮,控制台什么都没有打印。

二、QT中的TCP编程

1. TCP简介

  • TCP是面向连接的可靠的基于字节流的传输层通信协议。
  • TCP的服务端和客户端通信首先必须建立连接
  • 建立连接方式:服务端监听某个端口,当有客户端通过ip和port连接时,就会创建一个socket连接,之后就可以互发数据了。
  • QT中将socket视为输入输出流,数据的收发是通过read()write()来进行,而不是常见的send和recv。

----------------------------接下来,我们以一个实例来解析服务端和用户端程序编写-------------------------

2. 服务端程序编写

2.1 编写步骤:

【1】配置:①pro文件中添加network;②添加头文件<QTcpServer>和<QTcpSocket>。

【2】创建服务端对象:QTcpServer *tcpServer;(具体分配空间在构造函数中)

【3】服务端-客户端的信号槽连接:connect(tcpServer, SIGNAL(newConnection()), this, SLOT(mNewConnection()));

【4】编写【3】中的槽函数mNewConnection():

  • ①获取客户端对象:QTcpSocket *tmpTcpSocket = tcpServer->nextPendingConnection();
  • ②获取客户端信息:
    • 获取客户端ip:tmpTcpSocket->peerAddress().toString();
    • 获取客户端port:tmpTcpSocket->peerPort();
  • ③创建信号槽来处理客户端的连接状态:connect(tmpTcpSocket, SIGNAL(stateChanged(QAbstractSocket::SocketState)), this, SLOT(mStateChanged(QAbstractSocket::SocketState)));
  • ④创建信号槽来接收客户端发送的信息:connect(tmpTcpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));

【5】编写【4】中的槽函数mStateChanged(...):用switch-case结构来处理连接状态,当状态为断开连接时,删除当前调用的客户端对象。

【6】编写【4】中的槽函数receiveMessage():调用tmpTcpSocket->readAll()来获取客户端发送的信息。

【7】创建函数来给客户端发送数据:内部调用"客户端对象.write("写入的内容")"。

【8】开始监听:调用tcpServer->listen(QHostAddress("192.168.124.151"), 9999); 监听ip为192.168.124.151,port为9999。

2.2 编写代码:

【1】widget.h:

#ifndef WIDGET_H
#define WIDGET_H

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

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 mNewConnection();
    void receiveMessage();
    void mStateChanged(QAbstractSocket::SocketState socketState);
    void on_pushButton_3_clicked();
    void on_pushButton_clicked();
    void on_pushButton_2_clicked();

private:
    Ui::Widget *ui;
    QTcpServer *tcpServer;
};
#endif // WIDGET_H

【2】widget.cpp:

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


/***********************************************************
  * @函数名:Widget
  * @功  能:构造函数---创建服务端对象,与客户端连接
  * @参  数:parent---父对象
  * @返回值:无
  *********************************************************/
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    this->setWindowTitle("服务端");
    //创建对象,与客户端连接
    tcpServer = new QTcpServer(this);
    connect(tcpServer, SIGNAL(newConnection()), this, SLOT(mNewConnection()));
}


/***********************************************************
  * @函数名:~Widget
  * @功  能:析构函数
  * @参  数:无
  * @返回值:无
  *********************************************************/
Widget::~Widget()
{
    delete ui;
}


/***********************************************************
  * @函数名:mNewConnection
  * @功  能:槽函数---若客户端发起连接,服务端连接客户端
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::mNewConnection()
{
    //获取客户端
    QTcpSocket *tmpTcpSocket = tcpServer->nextPendingConnection();
    //打印客户端的ip和port
    QString ipAddr = tmpTcpSocket->peerAddress().toString();
    quint16 port = tmpTcpSocket->peerPort();
    ui->textBrowser->append("客户端的ip地址:" + ipAddr);
    ui->textBrowser->append("客户端的端口:" + QString::number(port));
    //处理客户端连接状态,接收客户端发送的数据
    connect(tmpTcpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));
    connect(tmpTcpSocket, SIGNAL(stateChanged(QAbstractSocket::SocketState)),
            this, SLOT(mStateChanged(QAbstractSocket::SocketState)));
}


/***********************************************************
  * @函数名:receiveMessage
  * @功  能:槽函数---服务端接收客户端发送的数据
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::receiveMessage()
{
    QTcpSocket *tmpTcpSocket = (QTcpSocket*)sender();
    ui->textBrowser->append("客户端:" + tmpTcpSocket->readAll());
}


/***********************************************************
  * @函数名:mStateChanged
  * @功  能:槽函数---服务端处理客户端的连接状态
  * @参  数:socketState---客户端连接状态
  * @返回值:无
  *********************************************************/
void Widget::mStateChanged(QAbstractSocket::SocketState socketState)
{
    QTcpSocket *tmpTcpSocket = (QTcpSocket*)sender();
    //处理状态,删除断开的QTcpSocket对象
    switch (socketState) {
    case QAbstractSocket::UnconnectedState://断开连接,删除对象
        ui->textBrowser->append("客户端断开连接");
        tmpTcpSocket->deleteLater();
        break;
    case QAbstractSocket::ConnectedState://已连接
        ui->textBrowser->append("客户端已连接");
        break;
    default:
        break;
    }
}


/***********************************************************
  * @函数名:on_pushButton_3_clicked
  * @功  能:按钮"发送消息"的槽函数,将文本信息发送给所有客户端
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_3_clicked()
{
    QList <QTcpSocket*> clients = tcpServer->findChildren<QTcpSocket*>();
    for (int i = 0; i < clients.length(); ++i) {
        clients[i]->write(ui->lineEdit->text().toUtf8());
    }
}


/***********************************************************
  * @函数名:on_pushButton_clicked
  * @功  能:按钮"开始监听"的槽函数,监听指定的ip和port
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_clicked()
{
    tcpServer->listen(QHostAddress("192.168.124.151"), 9999);
}


/***********************************************************
  * @函数名:on_pushButton_2_clicked
  * @功  能:按钮"停止监听"的槽函数
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_2_clicked()
{
    tcpServer->close();
}

3. 客户端程序编写

3.1 编写步骤:

【1】配置:①pro文件中添加network;②添加头文件<QTcpSocket>和<QHostAddress>。

【2】创建客户端对象:QTcpSocket *tcpSocket;(具体分配空间在构造函数中)

【3】创建信号槽来处理客户端的连接状态:connect(tcpSocket, SIGNAL(stateChanged(QAbstractSocket::SocketState)), this, SLOT(mStateChanged(QAbstractSocket::SocketState)));

【4】创建信号槽来接收客户端发送的信息:connect(tcpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));

【5】编写【3】中的槽函数mStateChanged(...):switch-case处理。

【6】编写【4】中的槽函数receiveMessage(): 调用tcpSocket->readAll().

【7】创建函数来给服务端发送数据:内部调用"客户端对象.write("写入的内容")"。

【8】启动连接服务端:调用tcpSocket->connectToHost(QHostAddress("192.168.124.151"), 9999);

【9】断开连接服务端:调用tcpSocket->disconnectFromHost();

3.2 编写代码:

【1】widget.h:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTcpSocket>
#include <QHostAddress>

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 receiveMessage();
    void mStateChanged(QAbstractSocket::SocketState socketstate);
    void on_pushButton_3_clicked();
    void on_pushButton_clicked();
    void on_pushButton_2_clicked();

private:
    Ui::Widget *ui;
    QTcpSocket *tcpSocket;
};
#endif // WIDGET_H

【2】widget.cpp:

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


/***********************************************************
  * @函数名:Widget
  * @功  能:构造函数---创建客户端对象,与服务端连接
  * @参  数:parent---父对象
  * @返回值:无
  *********************************************************/
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    //ui部分
    ui->setupUi(this);
    this->setWindowTitle("客户端");
    ui->pushButton->setEnabled(true);
    ui->pushButton_2->setEnabled(false);
    //tcp部分
    tcpSocket = new QTcpSocket(this);
    connect(tcpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));
    connect(tcpSocket, SIGNAL(stateChanged(QAbstractSocket::SocketState)),
            this, SLOT(mStateChanged(QAbstractSocket::SocketState)));
}


/***********************************************************
  * @函数名:~Widget
  * @功  能:析构函数
  * @参  数:无
  * @返回值:无
  *********************************************************/
Widget::~Widget()
{
    delete ui;
}


/***********************************************************
  * @函数名:receiveMessage
  * @功  能:槽函数---客户端接收服务端发送的数据
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::receiveMessage()
{
    ui->textBrowser->append("服务端:" + tcpSocket->readAll());
}


/***********************************************************
  * @函数名:mStateChanged
  * @功  能:槽函数---客户端连接状态改变的处理
  * @参  数:socketstate---当前连接状态
  * @返回值:无
  *********************************************************/
void Widget::mStateChanged(QAbstractSocket::SocketState socketstate)
{
    switch (socketstate) {
    case QAbstractSocket::UnconnectedState:
        ui->textBrowser->append("与服务端断开连接");
        ui->pushButton->setEnabled(true);
        ui->pushButton_2->setEnabled(false);
        break;
    case QAbstractSocket::ConnectedState:
        ui->textBrowser->append("与服务端成功连接");
        ui->pushButton->setEnabled(false);
        ui->pushButton_2->setEnabled(true);
        break;
    default:
        break;
    }
}


/***********************************************************
  * @函数名:on_pushButton_3_clicked
  * @功  能:"发送消息"按钮的槽函数,必须连接了服务端才发送
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_3_clicked()
{
    if (tcpSocket->state() == QAbstractSocket::ConnectedState){
        tcpSocket->write(ui->lineEdit->text().toUtf8());
    }
    else {
        ui->textBrowser->append("请先连接服务端!");
    }
}


/***********************************************************
  * @函数名:on_pushButton_clicked
  * @功  能:"连接服务端"按钮的槽函数
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_clicked()
{
    tcpSocket->connectToHost(QHostAddress("192.168.124.151"), 9999);
}


/***********************************************************
  * @函数名:on_pushButton_2_clicked
  * @功  能:"断开服务端"按钮的槽函数
  * @参  数:无
  * @返回值:无
  *********************************************************/
void Widget::on_pushButton_2_clicked()
{
    tcpSocket->disconnectFromHost();
}

4. 服务端与客户端测试

4.1 注意事项:

  • 服务端中的"开始监听"和"停止监听"应该设置成互斥的。方法一:在ui设计器里将它们添加到一个按钮组,然后选中"exclusive",将它们的"checkable"勾选上,再将"停止监听"按钮的"checked"勾选。方法二:使用代码ui->pushButton->setEnable(对/错);实现它们逻辑的互斥。
  • 客户端中的"连接服务端"和"断开服务端"应该用上述的方法二实现互斥。
  • 在服务端中,当客户端断开连接时,直接调用delete可能会出错(其他地方可能还在用这个变量)。因此,应当使用tmpTcpSocket.deleteLater(); 来删除客户端对象。

4.2 运行效果:


三、QT中的UDP编程

1. UDP简介

  • 是一个轻量级的,不可靠的,面向数据报的无连接协议。
  • 通常音频、视频和普通数据在传送时使用 UDP 较多。
  • UDP 消息传送有三种模式:单播、广播和组播三种模式。

2. UDP单播与广播程序

2.1 编写步骤:

【1】配置:①pro文件中添加network;②添加头文件<QUdpSocket>。

【2】创建UDP对象:QUdpSocket *udpSocket;(具体分配空间在构造函数中)

【3】编写函数绑定端口:调用udpSocket->bind(端口号);

【4】编写函数解除绑定:调用udpSocket->abort();

【5】“发送-接收”的信号槽连接:connect(udpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));

【6】编写【5】中的receiveMessage来获取接收的数据:内部调用udpSocket->readDatagram(mesg.data(), mesg.size(), &receiveAddress, &port);

【7】编写函数来单播发送数据:调用udpSocket->writeDatagram(ui->lineEdit->text().toUtf8(), QHostAddress("127.0.0.1"), 9999);

【8】编写函数来广播发送数据:与【7】类似,只是将倒数第二个参数换成QHostAddress::Broadcast。要广播几个ip和端口,就调用几次writeDatagram。

【8】要想获取当前连接的状态,操作和TCP中的一样。

2.2 实例代码:

【1】widget.h:

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QUdpSocket>

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 receiveMessage();

    void on_pushButton_3_clicked();

    void on_pushButton_clicked();

    void on_pushButton_2_clicked();

    void on_pushButton_4_clicked();

private:
    Ui::Widget *ui;
    QUdpSocket *udpSocket;
};
#endif // WIDGET_H

【2】widget.cpp:

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

/* 构造函数 */
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    ui->pushButton->setEnabled(true);
    ui->pushButton_2->setEnabled(false);
    //创建udp对象
    udpSocket = new QUdpSocket(this);
    connect(udpSocket, SIGNAL(readyRead()), this, SLOT(receiveMessage()));
}


/* 析构函数 */
Widget::~Widget()
{
    delete ui;
}


/* 接收信息的槽函数*/
void Widget::receiveMessage()
{
    QByteArray mesg;
    mesg.resize(udpSocket->pendingDatagramSize());
    QHostAddress receiveAddress;
    quint16 port;

    while (udpSocket->hasPendingDatagrams()) {
        udpSocket->readDatagram(mesg.data(), mesg.size(), &receiveAddress, &port);//保存接收的数据
        ui->textBrowser->append("接收来自:" + receiveAddress.toString() + ", 端口:" + QString::number(port));
        ui->textBrowser->append("接收信息:" + mesg);
    }
}


/* 发送信息按钮的槽函数 */
void Widget::on_pushButton_3_clicked()
{
    udpSocket->writeDatagram(ui->lineEdit->text().toUtf8(), QHostAddress("127.0.0.1"), 9999);
    ui->textBrowser->append("发送信息:" + ui->lineEdit->text().toUtf8());
}


/* 绑定端口按钮的槽函数 */
void Widget::on_pushButton_clicked()
{
    udpSocket->bind(9999);
    ui->textBrowser->append("当前绑定端口:" + QString::number(9999));
    ui->pushButton->setEnabled(false);
    ui->pushButton_2->setEnabled(true);
}


/* 解除绑定按钮的槽函数 */
void Widget::on_pushButton_2_clicked()
{
    udpSocket->abort();
    ui->textBrowser->append("解除绑定");
    ui->pushButton->setEnabled(true);
    ui->pushButton_2->setEnabled(false);
}


/* 广播信息按钮的槽函数 */
void Widget::on_pushButton_4_clicked()
{
    udpSocket->writeDatagram(ui->lineEdit->text().toUtf8(), QHostAddress::Broadcast, 9999);
    ui->textBrowser->append("发送信息:" + ui->lineEdit->text().toUtf8());
}

【3】ui设计器:

【4】运行效果:双开该程序,每次新运行程序前修改一下绑定的端口号并编译。同时,这两个端口号需要对应才能显示单播的效果。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

开启数字化校园解决方案,实现教育智能化

现代社会的教育面临诸多挑战&#xff0c;如何提高教育质量&#xff0c;实现教育智能化成为了当务之急。数字化校园解决方案应运而生&#xff0c;为学校提供了全新的教学模式和管理方式。本文将介绍数字化校园解决方案的重要性&#xff0c;以及如何开启数字化校园&#xff0c;实…

【端午安康,给大家讲个“网络”故事,深刻一下!】

牛马我&#x1f434;上周又挨锤了&#xff0c; 网络是不稳定的&#xff0c;博学多知的你可能知道&#xff0c;可能不知道。但假如没亲身经历过&#xff0c;知不知道都不深刻&#xff0c;牛马踩了个网络的坑&#xff0c;深刻了&#xff0c;这里分享下&#xff0c; 一个真相 无…

【Python报错】已解决ImportError: cannot import name ‘triu’ from ‘scipy.linalg’

成功解决“ImportError: cannot import name ‘triu’ from ‘scipy.linalg’”错误的全面指南 在Python编程中&#xff0c;尤其是在使用scipy这个科学计算库时&#xff0c;可能会遇到ImportError错误&#xff0c;提示无法从scipy.linalg模块中导入名为triu的函数。这个错误通…

深入JVM:线上内存泄漏问题诊断与处理

文章目录 深入JVM&#xff1a;线上内存泄漏问题诊断与处理一、序言二、内存泄漏概念三、内存泄漏环境模拟四、内存泄漏诊断与解决1、步骤一&#xff1a;获取堆内存快照文件&#xff08;1&#xff09;获取正在运行程序dump文件&#xff08;2&#xff09;获取已终止程序dump文件 …

HP Laptop 14s-fr1xxx原厂oem预装Win11系统ISO镜像下载

惠普星青春版14s-fr1xxx笔记本电脑原装出厂Windows11系统安装包&#xff0c;恢复出厂开箱状态一模一样 链接&#xff1a;https://pan.baidu.com/s/11Qe5XgCmH3emIVEpvoKclg?pwdm1qe 提取码&#xff1a;m1qe 适用型号&#xff1a;14s-fr1xxx 14s-fr0001AU、14s-fr0002AU、…

VMware Fusion 如何增加linux硬盘空间并成功挂载

文章目录 0. 前言1. 增加硬盘空间2. 硬盘分区2.1 查看硬盘2.2 分区2.3 格式化2.4 挂载 3. 参考 0. 前言 如果发现虚拟机分配的硬盘不足&#xff0c;需要增加硬盘空间。本文教给大家如何增加硬盘空间并成功挂载。 查看当前硬盘使用情况&#xff1a; df -h可以看到&#xff0c…

sqli-labs 靶场 less-7 第七关详解:OUTFILE注入与配置

SQLi-Labs是一个用于学习和练习SQL注入漏洞的开源应用程序。通过它&#xff0c;我们可以学习如何识别和利用不同类型的SQL注入漏洞&#xff0c;并了解如何修复和防范这些漏洞。Less 7 SQLI DUMB SERIES-7判断注入点 进入页面中&#xff0c;并输入数据查看结果。 发现空数据提…

求宇文玥在水下的浮力和赵丽颖捞他的时间

关注微信公众号 数据分析螺丝钉 免费领取价值万元的python/java/商业分析/数据结构与算法学习资料 2024年汉东省在达康书记的带领下率先实现高考试点改革。为让更多的考生能提升对他们的理解和记忆&#xff0c;把电视剧的场景融入考试题目中。确保学生看一遍就懂&#xff0c;想…

debian12安装时分区方案

一、初次尝试 一共设置了4个分区&#xff0c;其中根目录/分区46G&#xff0c;swap分区10G&#xff08;电脑内存为6G&#xff09;&#xff0c;/boot分区200M&#xff0c;/home分区55G。系统安装之后的实际占有情况为&#xff1a; 二、调整后情况 一共设置了4个分区&#xff0c…

基于R语言BIOMOD2 及机器学习方法的物种分布模拟与案例分析

原文链接&#xff1a;基于R语言BIOMOD2 及机器学习方法的物种分布模拟与案例分析https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247606139&idx4&snf94ec30bfb5fa7ac0320403d49db3b66&chksmfa821e9ccdf5978a44a9ba96f6e04a121c0bbf63beea0940b385011c0b…

Spring运维之boo项目表现层测试匹配响应执行状态响应体JSON和响应头

匹配响应执行状态 我们创建了测试环境 而且发送了虚拟的请求 我们接下来要进行验证 验证请求和预期值是否匹配 MVC结果匹配器 匹配上了 匹配失败 package com.example.demo;import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Auto…

【网络教程】Iptables官方教程-学习笔记7-简单理解IPTABLES规则的作用流程

前面学习了IPTABLES的所有功能介绍后&#xff0c;一个Linux设备里的IPTABLES规则集是如何运行的&#xff0c;这里简单做个介绍。 在Linux设备里输入"iptables -nvl",得到该设备的所有防火墙规则&#xff0c;得到的结果中可以看到这个设备防火墙里所有的链以及链里的…

2024年CKA模拟系统制作 | step-by-step | 1、基础环境准备

目录 一、软件环境 二、虚拟网络环境准备 1、编辑虚拟网络 2、网络设置 三、新建虚拟主机 1、新建目录 2、新建虚拟主机 四、系统安装 1、装载系统镜像 2、开启虚拟机 3、选择语言 4、键盘选择 5、网络配置 6、代理设置 7、设置软件源 8、存储设置 9、名称设置 …

计算机网络 —— 网络层 (路由协议)

计算机网络 —— 网络层 &#xff08;路由协议&#xff09; 什么是路由协议内部网关协议RIP关键特性 OSPF主要特点 外部网关协议BGP关键特性 我们今天来看路由协议&#xff1a; 什么是路由协议 路由协议是网络设备&#xff08;主要是路由器&#xff09;用来决定数据包在网络中…

【爬虫实战项目一】Python爬取豆瓣电影榜单数据

目录 一、环境准备 二、编写代码 2.1 分页分析 2.2 编码 一、环境准备 安装requests和lxml pip install requests pip install lxml 二、编写代码 2.1 分页分析 编写代码前我们先看看榜单的url 我们假如要爬取五页的数据&#xff0c;那么五个url分别是&#xff1a; htt…

工业互联网数字中台建设方案(ppt原件)

工业互联网数字中台解决方案旨在为企业提供全面、高效的数据驱动能力。该方案主要包括以下几个核心部分&#xff1a; 数据中台&#xff1a;作为核心&#xff0c;数据中台负责汇聚、整合、提纯和加工各类工业数据&#xff0c;实现数据资产的标准化、模型化和模块化。通过提供API…

cmake使用make和Ninja构建对比

前提 make和Ninja是两个常见的构建工具&#xff0c;在网上查阅了一些资料&#xff0c;说是Ninja比make构建速度要快很多。但是具体不知道快多少&#xff0c;所以趁着这次编译clang的机会&#xff0c;分享下它们在时间方面差多少。 步骤 下载llvm 参考llvm官网&#xff0c;这…

linux系统的网络工具和命令非常多,应该如何学习可以快速提高?

目录 一、linux的网络命令有多少&#xff1f; &#xff08;一&#xff09;网络配置 &#xff08;二&#xff09;网络监控 &#xff08;三&#xff09;网络诊断 &#xff08;四&#xff09;网络通信 &#xff08;五&#xff09;网络文件传输 &#xff08;六&#xff09;网…

java web:springboot mysql开发的一套家政预约上门服务系统源码:家政上门服务系统的运行流程

java web&#xff1a;springboot mysql开发的一套家政预约上门服务系统源码&#xff1a;家政上门服务系统的运行流程 家政上门服务系统的优势 服务质量更稳定&#xff1a;由专业的家政人员提供服务&#xff0c;经过严格的培训和筛选。 价格更透明&#xff1a;采用套餐式收费&…

Valgo,类型安全,表达能⼒强的go验证器

valgo 是一个为 Go 语言设计的类型安全、表达性强且可扩展的验证库。该库的特点包括&#xff1a; github.com/cohesivestack/valgo 类型安全&#xff1a;利用 Go 语言的泛型特性&#xff08;从 Go 1.18 版本开始支持&#xff09;&#xff0c;确保验证逻辑的类型安全。表达性&a…
最新文章