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

[node.js] [HTTP/S] 实现 requests 发起 HTTP/S/1.1/2.0 请求

node.js 使用 V8 引擎来编译运行 javascript 代码,与浏览器中的环境不同的是,node.js 不包含 DOM 和 BOM 模块。

本文使用 node.js 的官方库来实现一个简单的 requests() 函数,可以用来发送 HTTP/1.1 和 HTTP/2.0 的请求。有关 HTTP/1.1 和 HTTP/2.0 请参见往期的文章 HTTP 版本的演进 。

在 node.js http2 默认支持 keep-alive 连接,使用 http2 来发起 HTTP 请求需要我们自己来管理 client (TCP 连接)。

思路:创建一个 TCPConnection 类,用来保存 client 对象。创建一个 ConnectionPool 类,用来自动管理连接池,并定期清理长期没有请求的 client 对象。 requests() 函数可以控制 HTTP 的版本 HTTP/1.1 还是 HTTP/2.0,也可以选择 GET 或者 POST 方法,接收的数据经过解压缩后和响应头一起封装到对象中进行返回。

代码如下:(代码中加入了足够多的注释)

// requests.js
import http from 'node:http';
import https from 'node:https';
import http2 from 'node:http2';
import zlib from 'node:zlib';// http2.connect 创建的 client 实例维护者一个 TCP 连接
// 将 client 封装成类,实现 TCP 连接的复用
class TCPConnection {  constructor(client, origin) {  this.client = client;  this.origin = origin;  // 记录 client 的 origin: https://example.com:portthis.expires = Date.now() + 2 * 60 * 1000; // 设置过期时间为2分钟后  // 监听 error 事件,打印错误信息,关闭 client this.client.on('error', (err) => {  console.error(`Client error for origin ${this.origin}:`, err);  this.close(); // 在出现错误时关闭连接  });  // 监听 client 的 close() 事件,打印 originthis.client.on('close', () => {  console.log(`Connection closed for origin ${this.origin}`);  // 连接关闭后,不需要再次关闭,因为 this.client.close() 已经被调用或者即将被调用  // 但可以更新任何相关的状态或日志  });  }isExpired() {  return Date.now() > this.expires;     // 返回 client 是否过期}  close() {  this.client.close();     // 用来关闭过期的 client,过多的 client 会消耗系统的资源}
} // 维护一个连接池,保存所有带有 client 的 TCPConnection 类的实例
class ConnectionPool {  constructor() {  this.connections = new Map(); // 使用 Map 来存储连接,以便按 origin 查找  this.checkInterval = setInterval(() => this.checkExpiredConnections(), 3 * 60 * 1000); // 每3分钟检查一次  }  // 添加一个带有 origin 的连接  addConnection(origin) {  const client = http2.connect(origin);  const connection = new TCPConnection(client, origin);  this.connections.set(origin, connection); // 使用 origin 作为键  return connection;  }  // 根据 origin 获取连接  getConnection(origin) {  let connection = this.connections.get(origin);  if (connection && !connection.isExpired()) {  console.log(`使用现有的连接: ${connection.origin}`);      // 测试是否使用了现有的连接connection.expires = Date.now() + 2 * 60 * 1000;     // 连接被重新使用,重置 expires 过期时间return connection.client; // 返回现有的 client  } else {  // 如果连接不存在或已过期,则创建新连接  connection = this.addConnection(origin);  return connection.client;  }  }  checkExpiredConnections() {  for (const [origin, connection] of this.connections) {  if (connection.isExpired()) {  connection.close(); // 关闭过期的连接  this.connections.delete(origin); // 从 Map 中移除  }  }  }  closeAll() {     // 程序结束后调用,关闭所有的连接for (const connection of this.connections.values()) {  connection.close();  }this.connections.clear(); // 清空 Map  clearInterval(this.checkInterval); // 清除定期检查  }  
}const conPool = new ConnectionPool();/*** 解压缩数据* @param {Buffer} data - 要解压缩的数据* @param {string} encoding - 数据的编码方式* @returns {Promise<Buffer>} - 解压缩后的数据*/
async function decompressData(data, encoding) {return new Promise((resolve, reject) => {switch (encoding) {case 'gzip':zlib.gunzip(data, (err, decoded) => {if (err) reject(err);else resolve(decoded);});break;case 'deflate':zlib.inflate(data, (err, decoded) => {if (err) reject(err);else resolve(decoded);});break;case 'br':zlib.brotliDecompress(data, (err, decoded) => {if (err) reject(err);else resolve(decoded);});break;/*case 'zstd':break;     //以后实现*/default:resolve(data); // 如果内容未经任何编码或者压缩,亦或者是图片、视频,直接返回原始数据}});
}/*** 调用前确定好 http 的版本,向 http/1.1 的服务器发送 2.0 的请求会报 Protocol Error 的错误。* @param {string} url - 想要请求的 url* @param {string} method - 想要使用的方法 'GET' / 'POST' 等* @param {string} httpVersion -  控制 http 的版本: ['1.1' | '2.0']* @param {object} headers - request headers 请求头* @param {Buffer|string} [data] - 适用于 POST 方法(可选)* @returns {Promise<{data: Buffer, headers: object}>} - 返回响应数据和 headers 的 Promise*/
export default async function requests(url, method, httpVersion, headers, data) {return new Promise((resolve, reject) => {0try {const reqUrl = new URL(url);/*const myURL = new URL('https://example.com:8080/path?query=param#hash');// 访问各个部分console.log(myURL.origin);   // "https://example.com:8080"console.log(myURL.protocol); // "https:"console.log(myURL.hostname);  // "example.com"console.log(myURL.port);      // "8080"console.log(myURL.pathname);  // "/path"console.log(myURL.search);    // "?query=param"console.log(myURL.hash);      // "#hash"*///console.log(reqUrl);if ( httpVersion === '1.1' ) {     // 使用 http/1.1 发出请求,node:http、node:https 版本都是 1.1const options = {hostname: reqUrl.hostname,port: reqUrl.port || (reqUrl.protocol === 'http:' ? 80 : 443),method: method,path: `${reqUrl.pathname}${reqUrl.search}`,headers: headers};//console.log(options);const request = (reqUrl.protocol === 'http:' ? http : https).request(options, (response) => {var resData = Buffer.alloc(0);     //创建一个大小为 0 字节的 Buffer 实例response.on('data', (chunk) => {resData = Buffer.concat([resData, chunk]);   // 将resData和新的数据块chunk合并});response.on('end', () => {decompressData(resData, response.headers['content-encoding'])    // 检索响应头的 content-encoding 字段进行响应的解码操作.then((decodedData) => {resolve({ data: decodedData, headers: response.headers });// 如果数据经过了 gzip / deflate / br 压缩,就执行解压操作再返回// 数据 和 响应头 封装到对象中一起返回,返回响应头的必要性:方便后续的查看响应头,以进行一些操作}).catch((err) => {reject(`Decompression error: ${err.message}`);});});});request.on('error', (e) => {reject(`Problem with request: ${e.message}`);});if (method === 'POST' && data) {// Ensure the data is a Buffer or convert itrequest.write(Buffer.isBuffer(data) ? data : Buffer.from(data));}request.end();} else if ( httpVersion === '2.0' ) {const client = conPool.getConnection(reqUrl.origin);/*const client = http2.connect(reqUrl.origin);http2.connect 会为每个连接创建新的 TCP 连接。如果想要重用连接,需要使用相同的 client 实例。http2 不允许 connection: keep-alive 。 因为 http2 本身就设计为支持持久连接的。这意味着,HTTP/2 连接自动保持打开状态,以便在同一连接上处理多个请求和响应。*/client.on('error', (err)=> { reject(`Create client error: ${err}`);});const options = {':authority': reqUrl.host,':method': method,':path': `${reqUrl.pathname}${reqUrl.search}`,':scheme': reqUrl.protocol.slice(0, -1), // 去掉末尾的 ':'...headers   // headers 所有属性合并到 options 中};/*// 将 headers 中的属性合并到 options 中,除了 ...headers,也可以使用以下语句:Object.assign(options, headers);*///console.log(options);const request = client.request(options);var resHeaders;request.on('response', (headers) => {resHeaders = headers;});var resData = Buffer.alloc(0);request.on('data', (chunk) => {resData = Buffer.concat([resData, chunk]);});request.on('end', () => {decompressData(resData, resHeaders['content-encoding']).then((decodeData) => {resolve({ data: decodeData, headers: resHeaders});}).catch((err) => {reject(`Decompression error: ${err.message}`);});// client.close(); 使用 ConnectionPool 类来管理 client/*当所有请求完成后,调用 client.close() 以关闭连接。只有在不再需要连接时才应该关闭。client.close(); 会关闭 TCP 连接,新的 http 请求将无法重用 TCP 连接。为了有效地重用连接,应保持 client 对象的引用,以便后续请求可以使用同一连接。可以在一个更大的作用域中定义 client,并在多个请求中复用。*/});request.on('error', (e) => {reject(`Problem with request: ${e.message}`);// client.close();});if (method === 'POST' && data) {request.write(Buffer.isBuffer(data) ? data : Buffer.from(data));}request.end();} else {reject(`Unsupported http version ${httpVersion}`);}} catch (error) {reject(`Invalid URL: ${error.message}`);};});
}process.on('exit', () => {console.log("执行清理工作: 清理所有的 client 连接 ...");conPool.closeAll();
});

在 app.js 中用来发起请求测试,请求图片,和请求我之前发布一段命令行下旋转 cube 的视频:

import requests from './requests/requests.js';
//app.js
//测试
import fs from 'fs';//请求图片并写入文件
requests('https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png', 'GET', "1.1", {}).then(resObj => {console.log(resObj.data);console.log(resObj.headers);fs.writeFile('baidu.png', resObj.data, {encoding: 'binary'}, error => {   //以二进制写入文件if (error) {console.error('写入文件时出错:', err);} else {console.log('文件已成功写入。');}});}).catch(error => {console.error("Error: ", error);});//请求一段视频
var videos = ['32b018315e66b2f02a2c08433b42fcc0_0.ts','32b018315e66b2f02a2c08433b42fcc0_1.ts','32b018315e66b2f02a2c08433b42fcc0_2.ts','32b018315e66b2f02a2c08433b42fcc0_3.ts','32b018315e66b2f02a2c08433b42fcc0_4.ts'
];
var baseUrl = "https://v-blog.csdnimg.cn/asset/999efa6d97215aa8905a1a05f7398e9f/play_video/";async function downloadAndWriteVideos() {for (let index = 0; index < videos.length; index++) {let url = baseUrl + videos[index];console.log(`index ${index}/${videos.length - 1} : requesting Url: ${url}.`);try {const resObj = await requests(url, 'GET', "2.0", {});//写入每个视频段到单独的文件await fs.promises.writeFile('cube_' + index + '.ts', resObj.data, { encoding: 'binary' });console.log('数据写入成功: cube_' + index + '.ts');// 追加到 cube.ts 文件await fs.promises.appendFile('cube.ts', resObj.data, { encoding: 'binary' });console.log('数据追加成功。');} catch (error) {console.error("Error: ", error);}}
}//调用函数
downloadAndWriteVideos();process.on('SIGINT', () => {console.log("接收到 SIGINT 信号,程序即将退出 ...");process.exit();
});process.on('SIGTERM', () => {console.log("接收到 SIGTERM 信号,程序即将退出 ...");process.exit();
});

相关文章:

[node.js] [HTTP/S] 实现 requests 发起 HTTP/S/1.1/2.0 请求

node.js 使用 V8 引擎来编译运行 javascript 代码&#xff0c;与浏览器中的环境不同的是&#xff0c;node.js 不包含 DOM 和 BOM 模块。 本文使用 node.js 的官方库来实现一个简单的 requests() 函数&#xff0c;可以用来发送 HTTP/1.1 和 HTTP/2.0 的请求。有关 HTTP/1.1 和 …...

基于Java Springboot线上约拍摄影预约微信小程序

一、作品包含 源码数据库全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui uniapp 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 微信开发者工…...

node.js基础学习-cheerio模块-简单小爬虫(五)

学习cheerio模块&#xff0c;简单做一个爬取图片网站的图片&#xff0c;并且将这些图片下载到本地指定的文件夹下&#xff0c;很多图片网站都有一些反爬取的机制&#xff0c;找的好几个都会报302错误&#xff0c;所以我找了一个小的图片网站&#xff0c;这个没有反爬取机制&…...

吾杯网络安全技能大赛WP(部分)

吾杯网络安全技能大赛WP(部分) MISC Sign 直接16进制解码即可 原神启动 将图片用StegSolve打开 找到了压缩包密码 将解出docx文件改为zip 找到了一张图片和zip 再把图片放到stegSlove里找到了img压缩包的密码 然后在document.xml里找到了text.zip压缩包密码 然后就出来fl…...

【python自动化一】pytest的基础使用

1.pytest简述 pytest‌ 是一个功能强大且灵活的Python测试框架&#xff0c;其主要是用于流程控制&#xff0c;具体用于UI还是接口自动化根据个人需要而定。且其具有丰富插件&#xff0c;使用时较为方便。咱们具体看下方的内容&#xff0c;本文按照使用场景展开&#xff0c;不完…...

使用 CFD 仿真进行阀门性能分析:第 II 部分

了解如何使用 Ansys Discovery 通过优化模式获得准确的阀门性能结果。 第 II 部分&#xff1a;优化模式下的模拟 阀门的模拟可以在 Explore &#xff08;探索&#xff09; 模式和 Refine &#xff08;优化&#xff09; 模式下执行。Explore 模式允许快速仿真&#xff0c;在长达…...

Node.js实现WebSocket教程

Node.js实现WebSocket教程 1. WebSocket简介 WebSocket是一种在单个TCP连接上提供全双工通信的协议&#xff0c;允许服务器和客户端之间进行实时、双向通信。本教程将详细讲解如何在Node.js中实现WebSocket。 2. 技术选型 我们将使用ws库来实现WebSocket服务器&#xff0c;…...

使用 Statsmodels 进行统计建模与分析

使用 Statsmodels 进行统计建模与分析 Statsmodels 是 Python 中一个功能强大的库&#xff0c;用于执行统计建模和计量经济学分析。它提供了一系列经典的统计模型和评估方法&#xff0c;涵盖线性回归、时间序列分析和广义线性模型等。 本文将带你深入了解 Statsmodels 的功能…...

【Linux】进程间通信

目录 一、管道 &#xff08;一&#xff09;概念 &#xff08;二&#xff09;匿名管道 1、概念 2、函数介绍 3、示例代码 4、原理 &#xff08;三&#xff09;命名管道 1、概念 2、函数介绍 3、示例代码 4、原理 &#xff08;四&#xff09;管道的读写规则 &#x…...

Python 网络爬虫高级教程:分布式爬取与大规模数据处理

经过基础爬虫和进阶爬虫的学习&#xff0c;我们已经掌握了爬虫的基本原理、动态内容处理及反爬机制的应对。然而&#xff0c;当我们面对海量数据或需要高效爬取多个站点时&#xff0c;分布式爬虫和数据存储、处理能力就显得尤为重要。本篇博客将带你迈向网络爬虫的高级阶段&…...

猫爪背后的情感密码

当家中那只可爱的猫咪时不时用它的小爪子轻拍我们时&#xff0c;很多人或许只当作是调皮捣蛋&#xff0c;实则背后大有深意。 猫用爪子打&#xff0c;可能是在向我们发出玩耍的邀约。在猫咪的天性里&#xff0c;捕猎本能根深蒂固。它们在幼年时与同伴的嬉戏打闹&#xff0c;便…...

【自用】管材流转项目前端重部署流程 vue2 webpackage4 vuecli4

一、配置 1.下载项目&#xff0c;使用 IDEA 打开&#xff0c;并配置 Nodejs 它提示我&#xff0c;需要 Node.js&#xff0c;因为 nodejs 14 的 installer 已经官网已经找不到了&#xff0c;使用 fnm 又太麻烦&#xff0c; 所以直接采用在 IDEA 中下载的方式就好了。 2.清除缓…...

关于c的子进程 fork()

fork() 是一个非常重要的系统调用&#xff0c;用于在 Unix-like 操作系统中创建一个新的进程。它会将当前进程&#xff08;父进程&#xff09;复制成一个新的进程&#xff08;子进程&#xff09;。子进程会从父进程的代码处继续执行&#xff0c;但具有不同的进程 ID。 fork() …...

耀圣控制设备有限公司:优质压滤机阀门的引领者

耀圣控制设备有限公司&#xff1a;优质压滤机阀门的引领者 在压滤机阀门领域&#xff0c;耀圣控制设备有限公司以其卓越的品质和领先的技术&#xff0c;成为了行业内备受瞩目的品牌。 耀圣控制设备有限公司专注于压滤机阀门的研发与生产&#xff0c;凭借着先进的工艺和严格的质…...

【C语言】结构体(四)

本篇重点是typedef关键字 一&#xff0c;是什么&#xff1f; typedef用来定义新的数据类型&#xff0c;通常typedef与结构体的定义配合使用。 简单来说就是取别名 ▶ struct 是用来定义新的数据类型——结构体 ▶ typedef是给数据类型取别名。 二&#xff0c;为什么&#xf…...

面向源代码的软件可信度量模型 T_{na}

面向源代码的软件可信度量模型 T n a T_{na} Tna​ 课程&#xff1a;软件质量分析 作业 可编写下面的java程序&#xff1a; package org.example;public class SourceCodeOrientedModel {public static void main(String[] args) {int total 41;int[] m {9, 22, 9, 5, 7, 1…...

Java11使用JVM同一日志框架启用日志记录

你可以使用-Xlog选项配置或启用Java虚拟机同一日志框架的日志记录。 -Xlog:gc*trace:file/Users/xx/gc-%t.log:time,tags,level,pid,tid,hostname,path:filecount3,filesize10K -Xlog:gc*trace:stdout:time,tags,level,pid,tid,hostname:filecount3,filesize10K -Xlog:gc*trac…...

k8s容器存储接口 CSI 相关知识

容器存储接口 CSI 相关知识 参考&#xff1a; https://blog.csdn.net/lovely_nn/article/details/122880876 https://developer.aliyun.com/article/783464 https://www.cnblogs.com/varden/p/15139819.html存储商需实现 CSI 插件的 NodeGetVolumeStats 接口&#xff0c;Kube…...

JDBC相关

请解释一下 JDBC 是什么&#xff1f; JDBC&#xff08;Java Database Connectivity&#xff09;是 Java 语言访问数据库的标准 API。它提供了一套统一的接口&#xff0c;使得 Java 程序能够与各种不同的数据库进行交互。 请说明一下 JDBC 连接池的使用方法和优势。 使用方法…...

Github提交Pull Request教程 Git基础扫盲(零基础易懂)

1 PR是什么&#xff1f; PR&#xff0c;全称Pull Request&#xff08;拉取请求&#xff09;&#xff0c;是一种非常重要的协作机制&#xff0c;它是 Git 和 GitHub 等代码托管平台中常见的功能&#xff0c;被广泛用于参与社区贡献&#xff0c;从而促进项目的发展。 PR的整个过…...

【计算机网络】实验7:默认路由和特定主机路由以及路由环路问题

实验 7&#xff1a;默认路由和特定主机路由以及路由环路问题 一、 实验目的 了解默认路由以及特定主机路由。 了解静态路由配置错误导致的路由环路问题。 二、 实验环境 • Cisco Packet Tracer 模拟器 三、 实验内容 1、默认路由以及特定主机路由 (1) 第一步&#xff…...

RK3506 PINCTRL学习

引脚命名规则 Rockchip Pin 的 ID 按照 控制器 (bank) 端口 (port) 索引序号 (pin) 组成。 GPIO&#xff08;通⽤输⼊输出&#xff09; 控制器和GPIO控制器数量一致&#xff1b;端口固定A/B/C/D索引序号固定为0/1/2/3/4/5/6/7 每个控制器可以控制32个IO&#xff0c;作为GPIO功…...

NS4828 0.8A 线性同步移动电源管理芯片

1 特性 ● 内置固定 0.8A 的线性充电模式 ● 涓流/恒流/恒压三段式充电&#xff0c;支持 0V 电池充电 ● 充电输入端有防反灌功能&#xff0c;不需要防反灌二极管 ● 0.8A同步升压转换器 ● 同步放电固定 5.1V 输出 ● 双灯充放电 LED 灯指示 ● 支持自动负载检测 ● 双灯充电与…...

重生之学C++篇—算术操作符

一、算术操作符 在写代码的时候&#xff0c;一定会涉及到计算。为了方便运算&#xff0c;C提供了一系列的操作符&#xff0c;其中有一组操作符叫做算术操作符。分别是&#xff1a; 、 - 、 * 、 / 、 %&#xff0c;这些操作符都是双目操作符&#xff08;有两个操作数&#xff…...

人机交互革命,为智能座舱市场激战注入一针「催化剂」

从AIGC到AGI赋能&#xff0c;智能座舱人机交互体验迎来新范式。 不断训练、迭代的大模型&#xff0c;为智能座舱带来了更全面的感知能力、更准确的认知理解&#xff0c;以及更丰富的交互模态&#xff0c;显著提升了其智能化水平。 “AI大模型的快速应用与迭代&#xff0c;推动…...

《只狼》运行时提示“mfc140u.dll文件缺失”是什么原因?“找不到mfc140u.dll文件”要怎么解决?教你几招轻松搞定

《只狼》运行时提示“mfc140u.dll文件缺失”的科普与解决方案 作为一名软件开发从业者&#xff0c;在游戏开发和维护过程中&#xff0c;我们经常会遇到各种运行时错误和系统报错。今天&#xff0c;我们就来探讨一下《只狼》这款游戏在运行时提示“mfc140u.dll文件缺失”的原因…...

VTK中矩阵vtkMatrix4x4类的介绍和使用

1、矩阵-齐次坐标介绍 常见的点一般是Pt&#xff08;X,Y,Z&#xff09;&#xff0c;相当于一个13矩阵&#xff0c;而矩阵相乘的话一般是第一个矩阵的列数要等于第二个矩阵的行数。此处需要引入齐次坐标的概念&#xff1a;从广义上讲&#xff0c;齐次坐标就是用n1维向量表示n 维…...

STM32的OTA

STM32的OTA&#xff08;Over-The-Air&#xff09;是一种通过无线通信方式&#xff0c;为设备分发新软件、配置甚至更新加密密钥的技术。以下是对STM32 OTA的详细介绍&#xff1a; 一、OTA升级概述 OTA升级允许中心位置向所有用户发送更新&#xff0c;确保每个接收者都无法拒绝…...

【超图】iClient3D for Cesium 以动静结合方式加载WMTS服务

作者&#xff1a;taco 一、问题来源 在最近支持的项目中&#xff0c;我们面临一个挑战&#xff1a;客户需要在前端动态加载高达3亿级别的白模底面数据。这样做的主要原因是客户的数据库会频繁更新&#xff0c;因此我们需要采用动态加载的方式来确保用户界面能够实时反映最新的…...

2.STM32通信接口之SPI通信---SPI实战《精讲》

SPI仅支持一主多从&#xff08;无应答机制&#xff09; 参照&#xff1a;《第十一部分》1.STM32通信接口之SPI通信---SPI介绍《精讲》-CSDN博客 在采用一主多从的模式下。从机未被选中&#xff0c;SN1时&#xff0c;从机的MISO会处于高阻态状态&#xff0c;SN0时&#xff0c;M…...

WangEditor — 一个轻量级富文本编辑器

官网&#xff1a;https://www.wangeditor.com/ 介绍 在现代 Web 开发中&#xff0c;富文本编辑器是一个常见的工具&#xff0c;它允许用户输入、编辑和格式化内容。在众多富文本编辑器中&#xff0c;WangEditor 是一个轻量级、高效且易于使用的选择。它支持 HTML 格式的内容编…...

Redis 之持久化

目录 介绍 RDB RDB生成方式 自动触发 手动触发 AOF&#xff08;append-only file&#xff09; Redis 4.0 混合持久化 Redis主从工作原理 总结 介绍 Redis提供了两个持久化数据的能力&#xff0c;RDB Snapshot 和 AOF&#xff08;Append Only FIle&#xff09;…...

Scala的模式匹配

package hfdobject Test34 {def main(args: Array[String]): Unit {//身份证val id "429005202023210042"// val id "319005202023210042"// val id "119005202023210042"//val id "339005202023210042"//截取前两位val pre id.sub…...

学习笔记052——Spring Boot 自定义 Starter

文章目录 Spring Boot 自定义 Starter1、自定义一个要装载的项目2、创建属性读取类 ServiceProperties3、创建 Service4、创建自动配置类 AutoConfigration5、创建 spring 工程文件6、将项目打成 jar 包7、jar 打包到本地仓库8、配置application.yml Spring Boot 自定义 Starte…...

react学习记录Day2

Materal UI组件 1、Typography 文字铸排 使用文字铸排可以尽可能清晰、高效地展示您的设计和内容。 2、useTranslation 在React应用程序中&#xff0c;国际化&#xff08;i18n&#xff09;通常是通过使用专门的库来实现的&#xff0c;其中一个流行的库是react-i18next。这个…...

4. STM32_定时器

概述 什么是定时器&#xff1a; 定时器核心就是计数器&#xff0c;是使用精准的时基&#xff0c;通过硬件的方式&#xff0c;实现定时功能的器件。 定时器的工作原理&#xff1a; 主频时钟CLK通过PSC进行分频后产生定时器时钟TIM CLK&#xff0c;计数器CNT根据TIM CLK的频率…...

JMeter中获取随机数、唯一ID、时间日期(包括当前日期增减)截取指定位数的字符等

在JMeter中&#xff0c;您可以使用内置的函数和一些额外的插件来获取随机数、唯一ID、时间日期以及截取指定位数的字符。以下是一些常用的方法&#xff1a; 获取随机数&#xff1a; 使用__Random函数&#xff0c;您可以在指定的最小值和最大值之间生成一个随机数。例如&#xf…...

Windows下从命令行(Powershell/CMD)发送内容到系统通知中心

Windows下从命令行&#xff08;Powershell/CMD&#xff09;发送内容到系统通知中心 01 前言 在平时写脚本的时候&#xff0c;将日志等信息直接输出到控制台固然是最直接的&#xff0c;而如果是一些后台执行的任务&#xff0c;不需要时刻关注运行细节但是又想知道一些大致的情…...

day01-搭建项目环境

1. 项目介绍 1.1 项目亮点 1.2 技术架构 2. 搭建环境 启动老师提供的虚拟机&#xff0c;里面常用的组件是开机启动的。 本地访问一些组件时需要手动敲写ip地址和组件端口&#xff0c;在本地的host文件中添加虚拟机的地址&#xff0c;而且虚拟机中有nginx反向代理&#xff0c…...

使用Tauri创建桌面应用

当前是在 Windows 环境下 1.准备 系统依赖项 Microsoft C 构建工具WebView2 (Windows10 v1803 以上版本不用下载&#xff0c;已经默认安装了) 下载安装 Rust下载安装 Rust 需要重启终端或者系统 重新打开cmd&#xff0c;键入rustc --version&#xff0c;出现 rust 版本号&…...

MySQL 函数创建中的 Err 1418:原因解析与解决指南20241203

&#x1f6a8; MySQL 函数创建中的 Err 1418&#xff1a;原因解析与解决指南 &#x1f4d6; 引言 在使用 MySQL 创建函数时&#xff0c;许多开发者会偶然遇到如下报错&#xff1a; [Err] 1418 - This function has none of DETERMINISTIC, NO SQL, or READS SQL DATA in its…...

GPT 1到4代的演进笔记

1. GPT-1 标题是 Improving Language Understanding by Generative Pre-Training. 发表于 2018.02, 比 bert(发布于 2018.10) 早了半年. 1.1 动机 困难:NLU 任务是多样的, 有 {textual entailment, question answering, semantic similarity assessment, document classifica…...

Zookeeper集群数据是如何同步的?

大家好&#xff0c;我是锋哥。今天分享关于【Zookeeper集群数据是如何同步的?】面试题。希望对大家有帮助&#xff1b; Zookeeper集群数据是如何同步的? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 Zookeeper集群中的数据同步是通过一种称为ZAB&#xff08;Zo…...

vue初始化脚手架

Vue初始化脚手架是指使用Vue CLI&#xff08;Command Line Interface&#xff09;工具来创建一个Vue项目的基础结构。Vue CLI是Vue.js官方提供的一个脚手架工具&#xff0c;它可以帮助开发者快速搭建Vue项目的基本框架&#xff0c;包括目录结构、配置文件、依赖管理等。 1. 初始…...

可编程网络在分布式深度学习通信瓶颈控制中的应用与未来展望

目录 可编程网络技术解决分布式深度学习通信瓶颈 1. 网络内聚合原语加速分布式深度学习 2. 流聚合与网络内数据处理设计 3. 可编程数据平面加速 4. 降低通信发生的频次 5. 使用大批量进行训练 可编程网络 可编程性定义 在大语言模型中的应用 举例说明: 可编程网络在…...

【教学类-43-25】20241203 数独3宫格的所有可能-使用模版替换(12套样式,空1格-空8格,每套510张,共6120小图)

前期做数独惨宫格的所有排列&#xff0c;共有12套样式&#xff0c;空1格-空8格&#xff0c;每套510张&#xff0c;共6120小图&#xff09; 【教学类-43-24】20241127 数独3宫格的所有可能&#xff08;12套样式&#xff0c;空1格-空8格&#xff0c;每套510张&#xff0c;共6120…...

组合问题变式——选数(dfs)

代码随想录听课笔记1——回溯算法-CSDN博客 这是从1&#xff0c;2&#xff0c;3...,n个数字中选出k个数的组合&#xff0c;输出组合的全部可能的代码 //组合&#xff1a;返回1-n中所有个数为k的组合 1,2,3,4 #include<bits/stdc.h> using namespace std; #define MAX 1…...

怎麼解決路由器IP地址衝突?

路由器IP地址衝突通常發生在網路中有兩個設備嘗試使用相同的IP地址時。這種衝突會導致網路連接問題&#xff0c;因為每個設備需要一個唯一的IP地址才能正常通信。 1. 重啟設備 重啟路由器和設備&#xff1a;有時候簡單的重啟可以解決問題&#xff0c;設備重新獲取一個新的IP地…...

【Flink】Flink 作业执行大致流程

Flink 作业执行流程 (Application 模式) 客户端通过 ClusterDescriptor 提交 Application 模式 Sql 任务到服务端&#xff0c;服务端调用作业时 StreamTableEnvironment 使用 FlinkSqlParser 将 SQL 转换为 Operation , StreamTableEnvironment 转换 SQL 过程中使用 CatalogSou…...

mdf文件数据处理之画图(subplots多信号展示同一张图中)

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…...