Node.js系列(5)--数据库操作指南
Node.js数据库操作指南 💾
引言
数据库操作是Node.js应用开发中的关键环节。本文将深入探讨Node.js数据库操作的实现方案,包括连接管理、查询优化、事务处理等方面,帮助开发者构建高效可靠的数据访问层。
数据库操作概述
Node.js数据库操作主要包括以下方面:
- 连接管理:连接池、故障恢复、负载均衡
- 查询处理:SQL构建、参数绑定、结果映射
- 事务管理:事务控制、隔离级别、一致性保证
- 性能优化:查询优化、缓存策略、批量操作
- 安全防护:SQL注入防护、访问控制、数据加密
数据库操作实现
数据库管理器
// 数据库管理器
class DatabaseManager {private static instance: DatabaseManager;private config: DatabaseConfig;private pools: Map<string, Pool>;private metrics: DatabaseMetrics;private constructor() {this.pools = new Map();this.metrics = {queries: 0,errors: 0,connections: 0,transactions: 0};this.config = {maxConnections: 10,idleTimeout: 30000,connectionTimeout: 5000,retryAttempts: 3};}// 获取单例实例static getInstance(): DatabaseManager {if (!DatabaseManager.instance) {DatabaseManager.instance = new DatabaseManager();}return DatabaseManager.instance;}// 初始化数据库管理器async init(config: DatabaseConfig): Promise<void> {this.config = { ...this.config, ...config };// 创建主数据库连接池await this.createPool('master', {host: config.masterHost,port: config.masterPort,database: config.database,user: config.user,password: config.password});// 创建从数据库连接池if (config.slaves) {for (const [name, slave] of Object.entries(config.slaves)) {await this.createPool(name, {host: slave.host,port: slave.port,database: config.database,user: config.user,password: config.password});}}// 设置事件监听器this.setupEventListeners();}// 创建连接池private async createPool(name: string, config: PoolConfig): Promise<void> {const pool = new Pool({...config,max: this.config.maxConnections,idleTimeoutMillis: this.config.idleTimeout,connectionTimeoutMillis: this.config.connectionTimeout});// 测试连接try {const client = await pool.connect();client.release();console.log(`Database pool ${name} created successfully`);} catch (error) {console.error(`Failed to create database pool ${name}:`, error);throw error;}this.pools.set(name, pool);}// 设置事件监听器private setupEventListeners(): void {for (const [name, pool] of this.pools.entries()) {pool.on('connect', () => {this.metrics.connections++;console.log(`New connection in pool ${name}`);});pool.on('error', error => {this.metrics.errors++;console.error(`Pool ${name} error:`, error);});}}// 执行查询async query<T>(sql: string,params: any[] = [],options: QueryOptions = {}): Promise<T> {const poolName = options.useReplica ? this.selectReplica() : 'master';const pool = this.pools.get(poolName);if (!pool) {throw new Error(`Database pool ${poolName} not found`);}let client;let retryCount = 0;while (retryCount < this.config.retryAttempts) {try {client = await pool.connect();const startTime = Date.now();const result = await client.query(sql, params);const duration = Date.now() - startTime;this.metrics.queries++;this.logQuery(sql, params, duration);return result.rows as T;} catch (error) {retryCount++;this.metrics.errors++;if (retryCount === this.config.retryAttempts) {throw new DatabaseError('Query failed after max retry attempts',sql,params,error);}await this.wait(Math.pow(2, retryCount) * 1000);} finally {if (client) {client.release();}}}throw new Error('Unexpected query execution path');}// 执行事务async transaction<T>(callback: (client: PoolClient) => Promise<T>): Promise<T> {const pool = this.pools.get('master');if (!pool) {throw new Error('Master database pool not found');}const client = await pool.connect();try {await client.query('BEGIN');this.metrics.transactions++;const result = await callback(client);await client.query('COMMIT');return result;} catch (error) {await client.query('ROLLBACK');throw error;} finally {client.release();}}// 批量插入async batchInsert<T>(table: string,records: T[],options: BatchOptions = {}): Promise<void> {const batchSize = options.batchSize || 1000;const batches = this.chunk(records, batchSize);for (const batch of batches) {const values = this.buildBatchValues(batch);const columns = Object.keys(batch[0]);const sql = `INSERT INTO ${table} (${columns.join(', ')})VALUES ${values}`;await this.query(sql);}}// 选择从库private selectReplica(): string {const replicas = Array.from(this.pools.keys()).filter(name => name !== 'master');if (replicas.length === 0) {return 'master';}// 简单轮询策略const index = Math.floor(Math.random() * replicas.length);return replicas[index];}// 构建批量插入值private buildBatchValues<T>(records: T[]): string {return records.map(record => {const values = Object.values(record).map(value => {if (value === null) return 'NULL';if (typeof value === 'string') return `'${value}'`;return value;}).join(', ');return `(${values})`;}).join(', ');}// 分割数组private chunk<T>(array: T[], size: number): T[][] {const chunks = [];for (let i = 0; i < array.length; i += size) {chunks.push(array.slice(i, i + size));}return chunks;}// 等待指定时间private wait(ms: number): Promise<void> {return new Promise(resolve => setTimeout(resolve, ms));}// 记录查询日志private logQuery(sql: string,params: any[],duration: number): void {console.log('Query executed:', {sql,params,duration: `${duration}ms`});}// 获取性能指标getMetrics(): DatabaseMetrics {return { ...this.metrics };}// 关闭所有连接池async close(): Promise<void> {for (const [name, pool] of this.pools.entries()) {await pool.end();console.log(`Database pool ${name} closed`);}}
}// 查询构建器
class QueryBuilder {private table: string;private conditions: string[] = [];private parameters: any[] = [];private orderClauses: string[] = [];private limitValue?: number;private offsetValue?: number;private joinClauses: string[] = [];constructor(table: string) {this.table = table;}// 添加条件where(column: string, operator: string, value: any): this {this.conditions.push(`${column} ${operator} $${this.parameters.length + 1}`);this.parameters.push(value);return this;}// 添加AND条件andWhere(column: string, operator: string, value: any): this {if (this.conditions.length > 0) {this.conditions.push('AND');}return this.where(column, operator, value);}// 添加OR条件orWhere(column: string, operator: string, value: any): this {if (this.conditions.length > 0) {this.conditions.push('OR');}return this.where(column, operator, value);}// 添加JOINjoin(table: string, condition: string): this {this.joinClauses.push(`JOIN ${table} ON ${condition}`);return this;}// 添加LEFT JOINleftJoin(table: string, condition: string): this {this.joinClauses.push(`LEFT JOIN ${table} ON ${condition}`);return this;}// 添加排序orderBy(column: string, direction: 'ASC' | 'DESC' = 'ASC'): this {this.orderClauses.push(`${column} ${direction}`);return this;}// 设置限制limit(value: number): this {this.limitValue = value;return this;}// 设置偏移offset(value: number): this {this.offsetValue = value;return this;}// 构建查询build(): { sql: string; params: any[] } {let sql = `SELECT * FROM ${this.table}`;if (this.joinClauses.length > 0) {sql += ' ' + this.joinClauses.join(' ');}if (this.conditions.length > 0) {sql += ' WHERE ' + this.conditions.join(' ');}if (this.orderClauses.length > 0) {sql += ' ORDER BY ' + this.orderClauses.join(', ');}if (this.limitValue !== undefined) {sql += ` LIMIT ${this.limitValue}`;}if (this.offsetValue !== undefined) {sql += ` OFFSET ${this.offsetValue}`;}return {sql,params: this.parameters};}
}// 数据库错误类
class DatabaseError extends Error {constructor(message: string,public sql: string,public params: any[],public originalError: Error) {super(message);this.name = 'DatabaseError';}
}// 接口定义
interface DatabaseConfig {masterHost: string;masterPort: number;database: string;user: string;password: string;maxConnections: number;idleTimeout: number;connectionTimeout: number;retryAttempts: number;slaves?: Record<string, SlaveConfig>;
}interface SlaveConfig {host: string;port: number;
}interface PoolConfig {host: string;port: number;database: string;user: string;password: string;
}interface QueryOptions {useReplica?: boolean;
}interface BatchOptions {batchSize?: number;
}interface DatabaseMetrics {queries: number;errors: number;connections: number;transactions: number;
}// 使用示例
async function main() {// 创建数据库管理器const dbManager = DatabaseManager.getInstance();// 初始化配置await dbManager.init({masterHost: 'localhost',masterPort: 5432,database: 'myapp',user: 'postgres',password: 'secret',maxConnections: 10,idleTimeout: 30000,connectionTimeout: 5000,retryAttempts: 3,slaves: {slave1: {host: 'slave1.example.com',port: 5432},slave2: {host: 'slave2.example.com',port: 5432}}});// 执行简单查询const users = await dbManager.query<User[]>('SELECT * FROM users WHERE age > $1',[18]);// 使用查询构建器const queryBuilder = new QueryBuilder('users').where('age', '>', 18).andWhere('status', '=', 'active').orderBy('created_at', 'DESC').limit(10);const { sql, params } = queryBuilder.build();const result = await dbManager.query(sql, params);// 执行事务await dbManager.transaction(async client => {await client.query('UPDATE users SET balance = balance - $1 WHERE id = $2', [100, 1]);await client.query('UPDATE users SET balance = balance + $1 WHERE id = $2', [100, 2]);});// 批量插入const newUsers = [{ name: 'User 1', age: 25 },{ name: 'User 2', age: 30 }];await dbManager.batchInsert('users', newUsers, { batchSize: 1000 });// 获取性能指标const metrics = dbManager.getMetrics();console.log('Database metrics:', metrics);// 关闭连接await dbManager.close();
}interface User {id: number;name: string;age: number;status: string;created_at: Date;
}main().catch(console.error);
最佳实践与建议
-
连接管理
- 使用连接池
- 配置最大连接数
- 设置连接超时
- 实现故障转移
-
查询优化
- 使用参数化查询
- 优化查询语句
- 合理使用索引
- 避免N+1查询
-
事务处理
- 合理设置隔离级别
- 控制事务范围
- 处理并发访问
- 实现事务补偿
-
性能优化
- 实现读写分离
- 使用批量操作
- 配置查询缓存
- 监控性能指标
总结
Node.js数据库操作需要考虑以下方面:
- 连接管理和故障处理
- 查询优化和性能调优
- 事务管理和数据一致性
- 安全防护和访问控制
- 监控和维护支持
通过合理的数据库操作实现,可以提高应用的性能和可靠性。
学习资源
- 数据库设计模式
- SQL优化技巧
- 事务处理机制
- 性能优化指南
- 安全最佳实践
如果你觉得这篇文章有帮助,欢迎点赞收藏,也期待在评论区看到你的想法和建议!👇
终身学习,共同成长。
咱们下一期见
💻
相关文章:
Node.js系列(5)--数据库操作指南
Node.js数据库操作指南 💾 引言 数据库操作是Node.js应用开发中的关键环节。本文将深入探讨Node.js数据库操作的实现方案,包括连接管理、查询优化、事务处理等方面,帮助开发者构建高效可靠的数据访问层。 数据库操作概述 Node.js数据库操…...
最大异或对 The XOR Largest Pair
题目来自洛谷网站: 思路: 两个循环时间复杂度太高了,会超时。 我们可以先将读入的数字,插入到字典树中,从高位到低位。对每个数查询的时候,题目要求是最大的异或对,所以我们选择相反的路径&am…...
简单介绍My—Batis
1.什么是My—Batis? My—Batis是一个持久层框架,提供了sql映射功能,能方便的将数据库表和java对象进行映射,通过My—Batis可以将项目中的数据存储在数据库中,以便我们进行调用。值得注意的是My—Batis和spring不是一回…...
案例分享|树莓派媒体播放器,重构商场广告的“黄金三秒”
研究显示,与传统户外广告相比,数字户外广告在消费者心中的记忆率提高了17%,而动态户外广告更是能提升16%的销售业绩,整体广告效率提升了17%。这一显著优势,使得越来越多资源和技术流入数字广告行业。 户外裸眼3D广告 无…...
硬件基础(5):(3)二极管的应用
文章目录 [toc]1. **整流电路****功能**:**工作原理**:**应用实例**:电路组成:整流过程:电路的应用: 2. **稳压电路****功能**:**工作原理**:**应用实例**:电路组成及功能…...
数据结构之栈的2种实现方式(顺序栈+链栈,附带C语言完整实现源码)
对于逻辑关系为“一对一”的数据,除了用顺序表和链表存储外,还可以用栈结构存储。 栈是一种“特殊”的线性存储结构,它的特殊之处体现在以下两个地方: 1、元素进栈和出栈的操作只能从一端完成,另一端是封闭的…...
vscode终端不识别npm 无法解析npm
vscode 用以管理员打开识别npm vscode 用普通用户打开不识别npm 刚换了一台新电脑,寻思安装各种环境,一顿操作猛如虎,当最后一个打开vscode后,运行项目发现,新建终端>npm run dev 无法识别。 在cmd 中 打node -…...
java泛型的协变、逆变和不变
引言 我们在看开源框架经常会看到泛型来定义数据类型。 有时候, <? extends T> 和 <? super T> 这样带通配符的泛型参数。 这种通配符的泛型是什么意思怎么用???? 概念 型变: 用来描述类…...
effective Java 学习笔记(第二弹)
effective Java 学习笔记(第一弹) 整理自《effective Java 中文第3版》 本篇笔记整理第3,4章的内容。 重写equals方法需要注意的地方 自反性:对于任何非空引用 x,x.equals(x) 必须返回 true。对称性:对于…...
Spring BeanFactoryPostProcessor 和 BeanPostProcessor有什么用?
BeanFactoryPostProcessor 和 BeanPostProcessor 都是 Spring 框架中非常重要的扩展点,我们在开发中可以利用 Spring 容器实例化 Bean、配置 Bean 以及初始化 Bean 的过程中进行自定义的干预。但它们的作用时机和作用对象不同。 1. BeanFactoryPostProcessor: 作用…...
Centos7 Gitlab17部署
确保你的安装源正常 1.安装依赖项 sudo yum install -y curl policycoreutils-python openssh-server openssh-clients postfix 2.安装Gitlab (1)添加仓库 curl https://packages.gitlab.com/install/repositories/gitlab/gitlab-ee/script.rpm.sh | sudo bash (2) 安装Gitl…...
一文解读DeepSeek在工业制造领域的应用
引言 在当今数字化浪潮席卷全球的背景下,各个行业都在积极寻求创新与变革,工业制造领域也不例外。然而,传统工业制造在生产效率、质量控制、成本管理等方面面临着诸多挑战。在这一关键时期,人工智能技术的兴起为工业制造带来了新的…...
基于动态 FOF(基金中的基金)策略的基金交易推荐系统的设计与实现思路
下面为你呈现一个基于动态 FOF(基金中的基金)策略的基金交易推荐系统的设计与实现思路,同时给出一个简单的 Python 示例代码。 系统设计 1. 需求分析 收集各类基金的历史数据,涵盖净值、收益率、风险指标等。依据动态 FOF 策略…...
第一次程序Hello Python
Python环境安装 安装地址 安装地址 https://www.python.org/ 1. 安装Python 下载完成后,双击安装包(如python-3.13.2.exe),按照提示进行安装。在安装过程中,确保勾选“Add Python to PATH”选项,以便在命…...
nvm 切换node 版本 但是没有带星号*
在 Windows 系统中配置 NVM_SYMLINK 环境变量的步骤如下: 1️⃣ 设置变量路径 变量名:NVM_SYMLINK 变量值:建议设置为 C:\Program Files\nodejs(需与后续步骤中 NVM 配置的符号链接路径一致) 2️⃣ 配置系统环境变量…...
Java定时任务的三重境界:从单机心跳到分布式协调
《Java定时任务的三重境界:从单机心跳到分布式协调》 本文将以生产级代码标准,揭秘Java定时任务从基础API到分布式调度的6种实现范式,深入剖析ScheduledThreadPoolExecutor与Quartz Scheduler的线程模型差异,并给出各方案的性能压…...
1.NextJS基础
NextJS注意要点 文件用来定义路由,folder name becomes the route name注意区分客户端渲染和服务器渲染 html渲染完成后给到客户端(此时网页内容已经全部提供),有利于crawler和优化seo逻辑更简单request deduplication减少API请求…...
晶晨/全志/联发科芯片系列电视盒子改固件包教程
声明:对电视盒子进行改包(修改固件包)是一项有一定技术门槛且存在风险的操作,可能会导致盒子变砖、失去保修等问题,同时私自修改固件可能违反相关法律法规和使用协议。以下为你提供一个通用的大致改包教程,…...
2025最新3个wordpress好用的主题
红色大气的wordpress企业主题,适合服务行业的公司搭建企业官方网站使用。是一款专为中小企业和个人开发者设计的WordPress主题,旨在提供专业的网站构建解决方案。 通过此WordPress主题,用户可以轻松创建和维护一个专业的企业网站,…...
GZCTF平台搭建及题目上传
前言 我用手里的Ubuntu虚拟机搭建的,大家根据自己的实际情况来吧 安装及部署 首先,你的虚拟机需要有Docker和Docker-Compose,前者可以看我之前的文章,另外一个可以输入下面的命令安装,注意先获取管理员权限ÿ…...
openGauss关联列数据类型不一致引起谓词传递失败
今天分享一个比较有意思的案例 注意:因为原始SQL很长,为了方便排版,简化了SQL 下面SQL跑60秒才出结果,客户请求优化 select dtcs.owner, dtcs.table_name, dtcs.column_name, dct.commentsfrom dba_tab_columns dtcsleft outer j…...
【网络安全基础学习】渗透测试工具--Burp Suite详细教程
Burp Suite与SniffMaster:网络安全工具的双剑合璧 Burp Suite(简称BP)是一款用于攻击web应用程序的集成平台。它包含了许多工具,并为这些工具设计了许多接口,以促进加快攻击应用程序的过程。然而,除了Burp…...
手机测试,工作中学习
要学习各种机型的截图方式、开发模式在哪。 荣耀机型:截图:关节快速敲两下。开发者模式在“系统和更新”里。 1.出现缺陷,需要获取日志。 学习adb生成日志:当测试中出现缺陷的,使用adb logcat -d > d:/log.txt …...
ctfshow WEB web2
1.查当前数据库名称 or 11 union select 1,database(),3 limit 1,2;#-- 得到数据库名称web2 2.查看数据库表的数量 or 11 union select 1,(select count(*) from information_schema.tables where table_schema web2),3 limit 1,2;#-- 得到数据库表数量为2 3.查表的名字 第…...
大疆上云api介绍
概述 目前对于 DJI 无人机接入第三方云平台,主要是基于 MSDK 开发定制 App,然后自己定义私有上云通信协议连接到云平台中。这样对于核心业务是开发云平台,无人机只是其中一个接入硬件设备的开发者来说,重新基于 MSDK 开发 App 工作量大、成本高,同时还需要花很多精力在无人…...
ASP.NET Web API + VUE3 整合阿里云OSS,后端API生成预签名上传Url,前端VUE进行上传
1、后端API 我用的是.net sdk6,所以先安装了这个Aliyun.OSS.SDK.NetCore 下面是后端生成上传Url的参考代码,主意request.ContentType,如果这里要是设置了,那么前端也要设置成一样的,如果前端是获取文件的contentType&…...
Java基础 3.22
1.break练习 //1-100之内的数求和,求当和第一次大于20的当前数i public class Break01 {public static void main(String[] args) {int n 0;int count 0;for (int i 1; i < 100; i) {count i;System.out.println("当前和为" count);if (count &g…...
如何快速解决 Postman 报错?
介绍一些 Postman 常见的报错与处理方法,希望能够对大家有所帮助。 Postman 一直转圈打不开的问题 Postman 报错处理指南:常见报错与解决方法...
软件性能效率测试工具有哪些?专业第三方软件检测机构推荐
在软件开发的新时代,软件性能效率测试已经成为每个企业不可或缺的一部分。无论是在竞争激烈的市场中,还是在追求卓越用户体验的过程中,都需要进行有效的性能测试。 一、软件性能效率测试的目标 1、响应时间:确保用户请求的响…...
手机销售终端MPR+LTC项目项目总体方案P183(183页PPT)(文末有下载方式)
资料解读:手机销售终端 MPRLTC 项目项目总体方案 详细资料请看本解读文章的最后内容。在当今竞争激烈的市场环境下,企业的销售模式和流程对于其发展起着至关重要的作用。华为终端正处于销售模式转型的关键时期,波士顿 - 华为销售终端 MPRLTC …...
redis使用
redis集群搭建可参考:redis集群搭建-CSDN博客 一、简述 Redis(Remote Dictionary Server)是一个开源的、基于内存的高性能键值对存储系统,通常用作数据库、缓存和消息代理。它支持多种数据结构,包括字符串、哈希、列…...
深入 SVG:矢量图形、滤镜与动态交互开发指南
1.SVG 详细介绍 SVG(Scalable Vector Graphics) 是一种基于 XML 的矢量图形格式,用于描述二维图形。 1. 命名空间 (Namespace) ★ 了解 命名空间 URI:http://www.w3.org/2000/svg 用途:在 XML 或 XHTML 中区分不同标…...
从双指针到单调栈,深挖“接雨水”的算法奥秘
从双指针到单调栈,深挖“接雨水”的算法奥秘 大家好,我是你们熟悉的算法领域大牛Echo_Wish。今天我们聊聊经典题目《接雨水》(Trapping Rain Water),不仅仅是讲解,而是深度对比两种高效解法:双…...
Labview和C#调用KNX API 相关东西
叙述:完全没有听说过KNX这个协议...................我这次项目中也是简单的用了一下没有过多的去研究 C#调用示例工程链接(labview调用示例在 DEBUG文件夹里面) 通过网盘分享的文件:KNX调用示例.zip 链接: https://pan.baidu.com/s/1NQUEYM11HID0M4ksetrTyg?pwd…...
Wireshark网络抓包分析使用详解
序言 之前学计网还有前几天备考华为 ICT 网络赛道时都有了解认识 Wireshark,但一直没怎么专门去用过,也没去系统学习过,就想趁着备考的网络相关知识还没忘光,先来系统学下整理点笔记~ 什么是抓包?抓包就是将网络传输…...
linux命令行工具进阶
文章目录 前言ssh免密登录,免密码登录,公私钥查看与修改IP地址临时修改永久修改 mount临时切换根文件系统永久切换根文件系统loop文件partedinitramfsuboot command line 前言 本文记录了一些不经常用到,但在某个时刻需要用到的一些指令。 免…...
【Linux文件IO】Linux中标准IO的API的描述和基本用法
Linux中标准IO的API的描述和基本用法 一、标准IO相关API1、文件的打开和关闭示例代码: 2、文件的读写示例代码:用标准IO(fread、fwrite)实现文件拷贝(任何文件均可拷贝) 3、文件偏移设置示例代码: 4、fgets fputs fget…...
【netstat和ss】Windows和Linux下的,网络连接排查简单案例
网络连接排查利器:netstat与ss命令详解 初识netstat:Windows下的网络连接查看工具 需要查看本机的网络连接情况时,Windows系统提供了一个非常实用的命令:netstat。和findstr组合形成一个有用的组合命令: netstat -a…...
【WPF】MVVM模式实现数据绑定以及Command事件绑定
1.引用类 using System.ComponentModel2.创建Command自定义类 public class DelegateCommand : ICommand{public bool CanExecute(object parameter){if (CanExecuteFunc null)return true;return this.CanExecuteFunc(parameter);}public event EventHandler CanExecuteChan…...
Flutter快速搭建聊天
之前项目中使用的环信聊天,我们的App使用的Flutter开发的 。 所以,就使用的 em_chat_uikit ,这个是环信开发的Flutter版本的聊天。 一开始,我们也用的环信的聊天,是收费的,但是,后面就发现&…...
网络层之IP协议
在讨论传输层时, 我们都只讨论了发送方和接收方的问题, 而没有讨论中间的网络形态的问题. 也就是数据包如何从主机传送到主机的? 如图, 主机B发送数据到主机C, 发送报文需要进行路径选择, 主机B-> F-> G-> H-> C-> D -> 主机C 这条路径是如何被选择出来的?…...
【设计模式】策略模式(Strategy Pattern)详解
策略模式(Strategy Pattern)详解 一、策略模式的定义 策略模式(Strategy Pattern)是一种行为型设计模式,它定义了一组算法,将每个算法封装起来,并使它们可以相互替换,从而让算法的…...
Elasticsearch:构建 AI 驱动的搜索体验
Elasticsearch 介绍 当你开始使用 Elastic 时,你将使用 Elasticsearch Relevance Engine™(ESRE),它专为 AI 搜索应用程序提供支持。借助 ESRE,你可以利用一整套开发者工具,包括 Elastic 的文本搜索、向量…...
数据文件误删除,OceanBase中如何重建受影响的节点
当不慎误删数据文件且当前没有现成的可替换节点时,在OceanBase中,不必急于采取极端措施,可以考虑运用 server_permanent_offline_time 参数,来重建受影响的节点。 原理: server_permanent_offline_time 是 OceanBase数…...
MySQL面试专题
1.什么是BufferPool? Buffer Pool基本概念 Buffer Pool:缓冲池,简称BP。其作用是用来缓存表数据与索引数据,减少磁盘IO操作,提升效率。 Buffer Pool由缓存数据页(Page) 和 对缓存数据页进行描述的控制块 组成, 控制…...
Redmi Note 11 T pro + 刷入 LinegaOs 22.1 记录 手机已经解锁bl.
Redmi Note 11 T pro 刷入 LinegaOs 22.1 记录 手机已经解锁bl. 获取LIneagaOS源码, 以及https://github.com/xiaomi-mediatek-devs 这个组织提供的代码,非常感谢 环境要求: ubuntu 22.04 需要准备的依赖 sudo apt install git curl vim…...
Python+Requests+Pytest+YAML+Allure接口自动化框架
GitHub源码地址(详细注释):源码 调试项目python自主搭建:附项目源码 一、项目介绍 本项目是基于 PythonRequestsPytestYAMLAllure 搭建的 接口自动化测试框架,用于对 REST API 进行测试。 框架的主要特点包括&#…...
如何解决Redis缓存异常问题(雪崩、击穿、穿透)
引言 Redis作为一种高性能的内存数据库,被广泛应用于缓存系统的构建中。然而,在实际应用过程中,我们常常会遇到三种典型的缓存异常问题:缓存雪崩、缓存击穿和缓存穿透。这些问题如果处理不当,可能会导致系统性能下降&…...
如何使用 Postman 进行接口测试?
使用 Postman 这一工具,可以轻松地进行接口测试。以下是一份简单的使用教程,帮助你快速上手。 Postman 接口测试教程:详细步骤及操作技巧...
记一次线上环境JAR冲突导致程序报错org.springframework.web.util.NestedServletException
一、问题描述 有个文件导入功能,用到了Hutool 的加密解密功能,本地运行完全可以,但是线上报错:“org.springframework.web.util.NestedServletException: Handler dispatch failed; nested exception is java.lang.NoClassDefFou…...