当前位置: 首页 > news >正文

(二) 用QWebSocket 实现服务端和客户端(详细代码直接使用)

目录

前言

一、服务器的代码:

1、服务器的思路

2、具体服务器的代码示例

二、客户端的代码:

1、客户端的思路(和服务器类似)

2、具体客户端的代码示例


前言

        要是想了解QWebSocket的详细知识,还得移步到上一篇文章:


WebSocket 详解,以及用QWebSocket 实现服务端和客户端(含代码例子)-CSDN博客WebSocket 详解,以及用QWebSocket 实现服务端和客户端(含代码例子)-CSDN博客

        本篇文章主要讲解如何利用QWebSocket 实现服务和客户之间的通讯

一、服务器的代码:

1、服务器的思路

(1)首先创建了一个服务器的基类,主要实现了服务类的基本接口:

1、创建服务器:new QWebSocketServer
2、监听:listen
m_pWebSocketServer->listen(QHostAddress::LocalHost, mPort);//端口号
3、有新的连接,收到这个信号:QWebSocketServer::newConnection
4、获得新的客户端:nextPendingConnection 
5、接收到信息时候,收到信号:QWebSocket::binaryMessageReceived 
6、断开连接,收到信号:QWebSocket::disconnected

注意:数据的接收和发送,有两种格式,二进制和文本的,具体按照实际需要的来选择;

(2)在服务器的基类上,封装一个具体使用的类,这个类,主要是添加了QThread,创建一个子线程来进行服务器的开启,监听和接收数据,不会影响主线程的事件。

2、具体服务器的代码示例

        接收和发送的数据,以二进制为例

(1)服务器基类:

服务器基类的头文件:

        1)开启一个端口号为“9000”的服务器

        2)监听的网路是:QHostAddress::Any

QHostAddress::Any表示服务端监听所有可用的网络接口。
它是一个特殊的IP地址,表示服务端可以接受来自任何IP地址的连接请求。
这通常用于在一个计算机上运行多个网络服务时,让服务端能够监听所有可用的网络接口,

以便接受来自不同网络接口的连接请求。

注意:也可以监听具体的IP地址:

例如:QHostAddress(strLocalHostIp)

#ifndef WEBSOCKETSERVERBASE_H
#define WEBSOCKETSERVERBASE_H#include <QObject>
#include <QtWebSockets>QT_FORWARD_DECLARE_CLASS(QWebSocketServer)
QT_FORWARD_DECLARE_CLASS(QWebSocket)
QT_FORWARD_DECLARE_CLASS(QString)class WebsocketServerBase : public QObject
{Q_OBJECT
public:explicit WebsocketServerBase(QString serverName,  quint16 port, QObject *parent = 0);virtual ~WebsocketServerBase();signals://客户端发来的数据void sigProcessServerMessage(const QByteArray &data);public slots://发送数据给客户端void slotSendToAllClients(const QByteArray &data);//启动websocket服务器void slotStartServer();private slots://处理新接入的连接void slotNewConnection();//处理链接断开的事件void slotSocketDisconnected();//接收数据,并转发void slotProcessBinaryMessage(const QByteArray &message);public://检测是否存在客户端bool hadClients();private:QWebSocketServer *m_pWebSocketServer = nullptr;QList<QWebSocket *> m_clients;unsigned short m_nPort = 9000;QString m_strServerName = "server";
};#endif // WEBSOCKETSERVERBASE_H

服务器基类的源文件:

#include "websocketserverbase.h"#include<QDebug>static QString getIdentifier(QWebSocket *peer)
{return QStringLiteral("%1:%2").arg(peer->peerAddress().toString(),peer->origin());
}WebsocketServerBase::WebsocketServerBase(QString serverName,  quint16 port, QObject *parent): QObject(parent),m_nPort(port),m_strServerName(serverName)
{
}WebsocketServerBase::~WebsocketServerBase()
{   if(m_pWebSocketServer){m_pWebSocketServer->close();//m_pWebSocketServer->abort();m_pWebSocketServer->deleteLater();}
}//接收到外部发来的信息,转发给客户端
void WebsocketServerBase::slotSendToAllClients(const QByteArray &data)
{qDebug() << __FUNCTION__;for (QWebSocket *pClient : qAsConst(m_clients)) {qDebug() << "data: " << data;pClient->sendBinaryMessage(data);}
}bool WebsocketServerBase::hadClients()
{return m_clients.size()>0;
}void WebsocketServerBase::slotStartServer()
{if(m_pWebSocketServer)return;m_pWebSocketServer = new QWebSocketServer(m_strServerName, QWebSocketServer::NonSecureMode, this);if (m_pWebSocketServer->listen(QHostAddress::Any, m_nPort)){connect(m_pWebSocketServer, &QWebSocketServer::newConnection, this, &WebsocketServerBase::slotNewConnection);qDebug() << "WebSocket is start, port:" << m_nPort;}
}void WebsocketServerBase::slotNewConnection()
{auto pSocket = m_pWebSocketServer->nextPendingConnection();QTextStream(stdout) << getIdentifier(pSocket) << " connected!\n";qDebug() << "client connected!";pSocket->setParent(this);//二进制数据的接收connect(pSocket, &QWebSocket::binaryMessageReceived, this, &WebsocketServerBase::slotProcessBinaryMessage);connect(pSocket, &QWebSocket::disconnected, this, &WebsocketServerBase::slotSocketDisconnected);m_clients << pSocket;
}void WebsocketServerBase::slotSocketDisconnected()
{QWebSocket *pClient = qobject_cast<QWebSocket *>(sender());QTextStream(stdout) << getIdentifier(pClient) << " disconnected!\n";if (pClient){m_clients.removeAll(pClient);pClient->deleteLater();}
}//接收客户端发来的数据,并转发出去
void WebsocketServerBase::slotProcessBinaryMessage(const QByteArray &data)
{qDebug() << __FUNCTION__ << " data:" << data;emit sigProcessServerMessage(data);//test//slotSendToAllClients(data);
}

(2)将服务器基类封装,改造下:(外面可以直接使用这个类进行通讯)

改造后服务器的头文件:

        此处本例是使用回调函数将结果抛给上一层调用者,在qt里,完全可以用信号槽代替的。

#ifndef READERWEBSOCKETSERVER_H
#define READERWEBSOCKETSERVER_H#include <QObject>
#include "Singleton.h"// 回调函数,将websocket的结果抛给上层
typedef void(*recvMsgToSerial)(const QByteArray &byteArray);class WebsocketServerBase;
class ReaderWebsocketServer : public QObject, public Singleton<ReaderWebsocketServer>
{Q_OBJECTfriend class Singleton<ReaderWebsocketServer>;public:explicit ReaderWebsocketServer(QObject *parent = 0);virtual ~ReaderWebsocketServer();public:// 设置回调函数void setCallBack(recvMsgToSerial pFunc, void* pUser = NULL);// 接收串口发来数据,转发给客户端bool sendData(const QByteArray &byteArray);signals://转发数据给客户端void sigSendToAllClients(const QByteArray &data);private slots://处理客户端发来的数据,转发给需要的地方void slotProcessServerMessage(const QByteArray &data);private:WebsocketServerBase* m_pWsServer = nullptr;QThread* m_thdWsServer = nullptr;void* m_pUser;							// 返回回调的对象recvMsgToSerial m_pRecvMsgToSerial;		// 回调
};#endif // READERWEBSOCKETSERVER_H

改造后服务器的源文件:

#include "readerwebsocketserver.h"#include <QThread>
#include <cstring>
#include "websocketserverbase.h"ReaderWebsocketServer::ReaderWebsocketServer(QObject *parent): QObject(parent)
{m_thdWsServer = new QThread();m_pWsServer = new WebsocketServerBase("reader", 9000);m_pWsServer->moveToThread(m_thdWsServer);connect(m_pWsServer, &WebsocketServerBase::sigProcessServerMessage, this, &ReaderWebsocketServer::slotProcessServerMessage);connect(this, &ReaderWebsocketServer::sigSendToAllClients, m_pWsServer, &WebsocketServerBase::slotSendToAllClients);connect(m_thdWsServer, &QThread::started, m_pWsServer, &WebsocketServerBase::slotStartServer);connect(m_thdWsServer, &QThread::finished, m_pWsServer, &WebsocketServerBase::deleteLater);connect(m_thdWsServer, &QThread::finished, m_thdWsServer, &WebsocketServerBase::deleteLater);m_thdWsServer->start();
}ReaderWebsocketServer::~ReaderWebsocketServer()
{if(m_thdWsServer){m_thdWsServer->quit();m_thdWsServer->wait();}if(m_pWsServer){m_pWsServer->deleteLater();}
}void ReaderWebsocketServer::setCallBack(recvMsgToSerial pFunc, void *pUser)
{if (nullptr != pFunc)m_pRecvMsgToSerial = pFunc;if (nullptr != pUser)m_pUser = pUser;
}//接收串口发来数据,转发给客户端
bool ReaderWebsocketServer::sendData(const QByteArray &byteArray)
{bool hadBnode = m_pWsServer->hadClients();if (hadBnode){emit sigSendToAllClients(byteArray);}
}//处理客户端发来的数据,转发给需要的地方
void ReaderWebsocketServer::slotProcessServerMessage(const QByteArray &byteArray)
{qDebug() << __FUNCTION__ ;m_pRecvMsgToSerial(byteArray);
}

二、客户端的代码:

1、客户端的思路(和服务器类似)

(1)首先创建了一个客户端的基类,主要实现了客户端的基本接口:

(2)在客户端基类上,封装一个具体使用的类:(外面可以直接使用这个类进行通讯)

        这个类,主要是添加了QThread 和QTimer,QThread 创建一个子线程来进行服务器的开启,监听和接收数据,不会影响主线程的事件;QTimer主要是发心跳包,实现断开重连机制;

2、具体客户端的代码示例

        接收和发送的数据,以二进制为例

(1)客户端基类:

客户端基类的头文件:

/** @Description: websocket客户端,用于与中间件通信*/#pragma once
#include <QObject>
#include <QByteArray>class QTimer;
class QWebSocket;
class WebSocketBase : public QObject
{Q_OBJECT
public:WebSocketBase(QObject *parent = nullptr);~WebSocketBase();void setWebSocketUrl(QString strUrl="");bool getConnectStatus();int RecvFrom(QByteArray& byteArray);signals:void sigClientBinaryMessageReceived(const QByteArray &byteArray); //借用websocket的信号函数public slots:void slotCreateDataRecWS();//创建websocket连接void slotSendBinaryMessageMessage(const QByteArray &byteArray);void slotReconnect();           /*-<周期重连函数 */void slotActiveReconnect();private slots:void slotOnConnected();                 /*-<socket建立成功后,触发该函数 */void slotOnBinaryMessageReceived(const QByteArray &byteArray);   /*-<收到Sev端的数据时,触发该函数 */void slotOnDisConnected();              /*-<socket连接断开后,触发该函数 */private:QWebSocket *m_pDataRecvWS;     /*-<websocket类 */QTimer *m_pTimer;            /*-<周期重连Timer */QString m_strURL;              /*连接URL*/bool m_bConnectStatus;         /*-<websocket连接状态,连接成功:true;断开:false */QByteArray m_byteArray;
};

客户端基类的源文件:

#include "WebSocketBase.h"
#include <QWebSocket>
#include <QTimer>
#include <QByteArray>
#include <cstring>WebSocketBase::WebSocketBase(QObject *parent) : QObject(parent),m_pDataRecvWS(nullptr),m_pTimer(nullptr),m_strURL(""),m_bConnectStatus(false),m_byteArray("")
{}WebSocketBase::~WebSocketBase()
{m_pTimer->stop();m_pTimer->deleteLater();m_pDataRecvWS->abort();m_pDataRecvWS->deleteLater();
}void WebSocketBase::setWebSocketUrl(QString strUrl)
{m_strURL = strUrl;if(m_strURL.isEmpty()){m_strURL = "127.0.0.1";}
}bool WebSocketBase::getConnectStatus()
{return m_bConnectStatus;
}int WebSocketBase::RecvFrom(QByteArray &byteArray)
{byteArray = m_byteArray;m_byteArray.clear();return byteArray.size();
}void WebSocketBase::slotCreateDataRecWS()
{if(nullptr == m_pTimer){m_pTimer = new QTimer();}qDebug() << "Server Address" << m_strURL;if(m_pDataRecvWS == nullptr){m_pDataRecvWS = new QWebSocket();connect(m_pDataRecvWS, &QWebSocket::disconnected, this, &WebSocketBase::slotOnDisConnected);connect(m_pDataRecvWS, &QWebSocket::binaryMessageReceived, this, &WebSocketBase::slotOnBinaryMessageReceived);connect(m_pDataRecvWS, &QWebSocket::connected, this, &WebSocketBase::slotOnConnected);connect(m_pTimer, &QTimer::timeout, this, &WebSocketBase::slotReconnect);m_pDataRecvWS->open(QUrl(m_strURL));}
}void WebSocketBase::slotSendBinaryMessageMessage(const QByteArray &message)
{if (m_pDataRecvWS)m_pDataRecvWS->sendBinaryMessage(message);
}void WebSocketBase::slotActiveReconnect()
{qDebug("try to Active Reconnect!!!");if(m_pDataRecvWS != nullptr){m_bConnectStatus = false;m_pDataRecvWS->abort();qDebug("Exec Active Reconnect!");m_pDataRecvWS->open(QUrl(m_strURL));}return;
}void WebSocketBase::slotReconnect()
{qDebug() << "try to reconnect:" << m_strURL;m_pDataRecvWS->abort();m_pDataRecvWS->open(QUrl(m_strURL));
}void WebSocketBase::slotOnConnected()
{qDebug("WebSocketBase websocket is already connect!");m_bConnectStatus = true;m_pTimer->stop();qDebug() << "Address:" << m_strURL;
}void WebSocketBase::slotOnDisConnected()
{qDebug() << "Address is disconnected:" << m_strURL;m_bConnectStatus = false;m_pTimer->start(3000);/*-<当连接失败时,触发重连计时器,设置计数周期为3秒 */
}void WebSocketBase::slotOnBinaryMessageReceived(const QByteArray& byteArray)
{m_byteArray = byteArray;
}

(2)将客户端基类封装,改造下:(外面可以直接使用这个类进行通讯)

改造后客户端头文件:

/** @Description: websocket客户端,用于与中间件通信*/
#pragma once#include <QObject>
#include <QCoreApplication>#include "Singleton.h"class WebSocketBase;
class QTimer;class WsReaderClient : public QObject, public Singleton<WsReaderClient>
{Q_OBJECTfriend class Singleton<WsReaderClient>;public:WsReaderClient(QObject *parent = nullptr);~WsReaderClient();public:void SendTo(const QByteArray &byteArray);int RecvFrom(QByteArray& byteArray);bool getConnectStatus();signals://转发数据给servervoid sigSendToServer(const QByteArray &byteArray);public slots://接收服务器数据void slotRecvServerData(const QByteArray &byteArray);//发送服务器心跳包void slotHeartBeatToServer();private:void readConfig();private:WebSocketBase* m_wsReaderClient;QThread* m_thdReaderClient;QTimer *m_pTimerReader;int m_nHeartBeatTimeOutReader;QString m_URL = "";
};

改造后客户端源文件:

#include "WsReaderClient.h"#include <QWebSocket>
#include <QTimer>
#include <QThread>
#include <QByteArray>
#include <cstring>
#include <QSettings>#include "WebSocketBase.h"WsReaderClient::WsReaderClient(QObject *parent): QObject(parent)
{readConfig();m_thdReaderClient = new QThread();m_wsReaderClient = new WebSocketBase();m_wsReaderClient->setWebSocketUrl(m_URL);m_wsReaderClient->moveToThread(m_thdReaderClient);connect(m_thdReaderClient, &QThread::started, m_wsReaderClient, &WebSocketBase::slotCreateDataRecWS);connect(this, &WsReaderClient::sigSendToServer, m_wsReaderClient, &WebSocketBase::slotSendBinaryMessageMessage);//connect(this, &WsReaderClient::sigReconnectServer, m_wsReaderClient, &WebSocketBase::slotActiveReconnect);connect(m_thdReaderClient, &QThread::finished, m_wsReaderClient, &WebSocketBase::deleteLater);connect(m_thdReaderClient, &QThread::finished, m_thdReaderClient, &QThread::deleteLater);m_thdReaderClient->start();m_pTimerReader = new QTimer(this);connect(m_pTimerReader, &QTimer::timeout, this, &WsReaderClient::slotHeartBeatToServer);//m_nHeartBeatTimeOutKeyBoard = 0;m_pTimerReader->start(10*1000);
}WsReaderClient::~WsReaderClient()
{m_pTimerReader->stop();m_pTimerReader->deleteLater();if(m_wsReaderClient){delete m_wsReaderClient;m_wsReaderClient = nullptr;}if(m_pTimerReader){delete m_pTimerReader;m_pTimerReader = nullptr;}
}void WsReaderClient::slotHeartBeatToServer()
{//todo
}void WsReaderClient::readConfig()
{// "/mnt/hgfs/SharedFiles/shanxi/Reader/linux_readerTest/bin/libReaderApi.so";QString appPath = QCoreApplication::applicationDirPath();qDebug() << "appPath=" << appPath;QString path = appPath + "/ReaderConfig.ini";QSettings settings(path, QSettings::IniFormat);m_URL = settings.value("Communication/ipAddr").toString();qDebug() << "m_URL=" << m_URL;
}void WsReaderClient::SendTo(const QByteArray &data)
{emit sigSendToServer(data);
}int WsReaderClient::RecvFrom(QByteArray &byteArray)
{return m_wsReaderClient->RecvFrom(byteArray);
}bool WsReaderClient::getConnectStatus()
{return m_wsReaderClient->getConnectStatus();
}

相关文章:

(二) 用QWebSocket 实现服务端和客户端(详细代码直接使用)

目录 前言 一、服务器的代码&#xff1a; 1、服务器的思路 2、具体服务器的代码示例 二、客户端的代码&#xff1a; 1、客户端的思路&#xff08;和服务器类似&#xff09; 2、具体客户端的代码示例 前言 要是想了解QWebSocket的详细知识&#xff0c;还得移步到上一篇文…...

关于我在配置zookeeper出现,启动成功,进程存在,但是查看状态却没有出现Mode:xxxxx的问题和我的解决方案

在我输入:zkServer.sh status 之后出现报错码. 报错码&#xff1a; ZooKeeper JMX enabled by default Using config: /opt/software/zookeeper/bin/../conf/zoo.cfgClient port found: 2181. Client address: localhost. Error contacting service. It is probably not runni…...

react及相关面试问题汇总

目录 1、什么是React&#xff1f;它的特点是什么&#xff1f; 2、解释一下虚拟DOM(Virtual DOM)的概念以及它的工作原理。 3、什么是组件(Component)&#xff1f;如何定义一个React组件&#xff1f; 4、什么是JSX&#xff1f;它与HTML的区别是什么&#xff1f;如何在React中…...

QT4到QT5移植出现的一些问题

转自&#xff1a;QT4到QT5移植出现的一些问题_西门子3gl qt5 许可证-CSDN博客 在上述作者基础上修改&#xff1a; 一、问题1&#xff1a;头文件的问题 1、QtGui/QApplication: No such file or directory 1.1错因 原因是Qt5源文件位置的改动 1.2解决 pro文件里&#xff0…...

【可解释AI】Alibi explain: 解释机器学习模型的算法

Alibi explain: 解释机器学习模型的算法 可解释人工智能简介Alibi特点算法Library设计展望参考资料 今天介绍Alibi Explain&#xff0c;一个开源Python库&#xff0c;用于解释机器学习模型的预测(https://github.com/SeldonIO/alibi)。该库具有最先进的分类和回归模型可解释性算…...

No191.精选前端面试题,享受每天的挑战和学习

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入…...

ROS基础—vscode创建工作空间

1、创建ROS工作空间 首先打开ubuntu的终端&#xff0c;接着依次输入如下的命令行&#xff1b; mkdir -p xxx_ws/src(必须得有 src) cd xxx_ws catkin_make当然我一般是新建一个叫做demo的工作空间&#xff0c;如 mkdir -p demo04_ws/src 2、启动vscode cd xxx_ws code . …...

机器学习复习(待更新)

01绪论 &#xff08;1&#xff09;机器学习基本分类&#xff1a; 监督学习&#xff08;有标签&#xff09;半监督学习&#xff08;部分标签&#xff0c;找数据结构&#xff09;无监督学习&#xff08;无标签&#xff0c;找数据结构&#xff09;强化学习&#xff08;不断交互&…...

taro(踩坑) npm run dev:weapp 微信小程序开发者工具预览报错

控制台报错信息&#xff1a; VM72:9 app.js错误: Error: module vendors-node_modules_taro_weapp_prebundle_chunk-JUEIR267_js.js is not defined, require args is ./vendors-node_modules_taro_weapp_prebundle_chunk-JUEIR267_js.js 环境&#xff1a; node 版本&#x…...

3. 深度学习——损失函数

机器学习面试题汇总与解析——损失函数 本章讲解知识点 什么是损失函数?为什么要使用损失函数?详细讲解损失函数本专栏适合于Python已经入门的学生或人士,有一定的编程基础。 本专栏适合于算法工程师、机器学习、图像处理求职的学生或人士。 本专栏针对面试题答案进行了优化…...

交叉编译 openssl

要在 x86 平台上编译适用于 aarch64 架构的 OpenSSL 动态库&#xff0c;你需要使用交叉编译工具链。可以按照以下步骤进行&#xff1a; 安装 aarch64 交叉编译工具链&#xff1a; $ sudo apt-get install gcc-aarch64-linux-gnu g-aarch64-linux-gnu 这将安装 aarch64 交叉编…...

C++文件的读取和写入

1、C对txt文件的读&#xff0c;ios::in #include<iostream> #include<fstream> using namespace std;int main() {ifstream ifs;ifs.open("test.txt",ios::in);if(!ifs.is_open()){cout<<"打开文件失败&#xff01;"<<endl;}char…...

住宅IP、家庭宽带IP以及原生IP,它们有什么区别?谷歌开发者账号应选择哪种IP?

IP地址&#xff08;Internet Protocol Address&#xff09;是互联网协议地址的简称&#xff0c;是互联网通信的基础&#xff0c;互联网上每一个网络设备的唯一标识符每个在线的设备都需要一个IP地址&#xff0c;这样才能在网络中找到它们并进行数据交换。 IP地址有很多种类型&…...

Linux内核分析(十三)--内存管理之I/O交换与性能调优

目录 一、引言 二、page cache ------>2.1、file-backed ------>2.2、匿名页(Anonymous page) ------>2.3、读写方式 ------>2.4、常驻内存 三、页面回收 ------>3.1、LRU算法 ------>3.2、嵌入式系统的zRAM 四、内存性能调优 ------>4.1、存储…...

前端使用webscoket

前端 <template><div class"wrap"><button click"socketEmit">连接Socket</button><button click"socketSendmsg">发送数据</button></div> </template><script> export default {data(…...

centos安装Git

一开始使用yum -y install git出来的版本比较低。 1.yum安装gityum -y install git 2.查看git的版本git --version 3.删除git&#xff08;版本过低&#xff09;yum remove -y git 后来通过源码编译后安装&#xff1a; 使用安装包安装&#xff1a; 1.下载安装包&#xff1a;h…...

网络编程 初探windows编程

目录 一、什么是Winodws编程 二、开发环境搭建以及如何学习 三、VA助手安装 四、第一个Win32程序 五、窗口类句柄/窗口类对象 六、Winodws消息循环机制 七、Windows数据类型 一、什么是Winodws编程 Windows 编程指的是在 Microsoft Windows 操作系统上进行软件开发的过…...

Vue3 ref函数和reactive函数

一、ref函数 我们在setup函数中导出的属性和方法虽然能够在模板上展示出来&#xff0c;但是并没有给属性添加响应式&#xff0c;因此&#xff0c;我们需要使用ref函数来为我们的数据提供响应式。 &#xff08;一&#xff09;引入ref函数 import { ref } from "vue"…...

docker常用命令详解

1. Image常见操作 (1)查看本地image列表 docker images docker image ls (2)获取远端镜像 docker pull (3)删除镜像[注意此镜像如果正在使用&#xff0c;或者有关联的镜像&#xff0c;则需要先处理完] docker image rm imageid docker rmi -f imageid docker rmi -f $(docker …...

采集Prestashop独立站采集Prestashop独立站

import java.net.URL 这一行导入了Java.net包中的URL类&#xff0c;这个类在处理URL链接时非常有用。 import org.jsoup.Jsoup 这一行导入了Jsoup库&#xff0c;它是一个强大的HTML和XML文档解析库&#xff0c;我们可以使用它来解析网页内容。 import org.jsoup.nodes.Docume…...

SpringCloudGateway 自定义局部过滤器

场景&#xff1a; 将所有请求转化为同一路径请求&#xff08;方便穿网配置&#xff09;在请求头内标识原来路径&#xff0c;然后在将请求分发给不同服务 AllToOneGatewayFilterFactory import lombok.Getter; import lombok.Setter; import lombok.extern.slf4j.Slf4j; impor…...

如何在最短时间内提升打ctf(web)的水平?

刚刚刷完2遍 bugku 的 web 题&#xff0c;前来答题。 每个人对刷题理解是不同&#xff0c;有的人是看了writeup就等于刷了&#xff0c;有的人是收藏了writeup就等于刷了&#xff0c;有的人是跟着writeup做了一遍就等于刷了&#xff0c;还有的人是独立思考做了一遍就等于刷了。…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程

本文较长&#xff0c;建议点赞收藏&#xff0c;以免遗失。更多AI大模型应用开发学习视频及资料&#xff0c;尽在聚客AI学院。 本文全面剖析RNN核心原理&#xff0c;深入讲解梯度消失/爆炸问题&#xff0c;并通过LSTM/GRU结构实现解决方案&#xff0c;提供时间序列预测和文本生成…...

基于matlab策略迭代和值迭代法的动态规划

经典的基于策略迭代和值迭代法的动态规划matlab代码&#xff0c;实现机器人的最优运输 Dynamic-Programming-master/Environment.pdf , 104724 Dynamic-Programming-master/README.md , 506 Dynamic-Programming-master/generalizedPolicyIteration.m , 1970 Dynamic-Programm…...

鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南

1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发&#xff0c;使用DevEco Studio作为开发工具&#xff0c;采用Java语言实现&#xff0c;包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...

人工智能(大型语言模型 LLMs)对不同学科的影响以及由此产生的新学习方式

今天是关于AI如何在教学中增强学生的学习体验&#xff0c;我把重要信息标红了。人文学科的价值被低估了 ⬇️ 转型与必要性 人工智能正在深刻地改变教育&#xff0c;这并非炒作&#xff0c;而是已经发生的巨大变革。教育机构和教育者不能忽视它&#xff0c;试图简单地禁止学生使…...

Netty从入门到进阶(二)

二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架&#xff0c;用于…...

push [特殊字符] present

push &#x1f19a; present 前言present和dismiss特点代码演示 push和pop特点代码演示 前言 在 iOS 开发中&#xff0c;push 和 present 是两种不同的视图控制器切换方式&#xff0c;它们有着显著的区别。 present和dismiss 特点 在当前控制器上方新建视图层级需要手动调用…...

[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】

大家好&#xff0c;我是java1234_小锋老师&#xff0c;看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】&#xff0c;分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...

Elastic 获得 AWS 教育 ISV 合作伙伴资质,进一步增强教育解决方案产品组合

作者&#xff1a;来自 Elastic Udayasimha Theepireddy (Uday), Brian Bergholm, Marianna Jonsdottir 通过搜索 AI 和云创新推动教育领域的数字化转型。 我们非常高兴地宣布&#xff0c;Elastic 已获得 AWS 教育 ISV 合作伙伴资质。这一重要认证表明&#xff0c;Elastic 作为 …...