uniapp 本地數據庫多端適配實例(根據運行環境自動選擇適配器)

項目有個需求,需要生成app和小程序,app支持離線數據庫,如果當前沒有網絡提醒用戶開啟離線模式,所以就隨便搞了下,具體的思路就是: 一個接口和多個實現類(類似后端的模板設計模式),例如sqlite實現類,indexedDB實現類等等,根據環境動態選用具體的實現類

indexedDB有些方法沒有改,sqlite沒有測試,此處就是提供一個思路,只是用來學習和研究著玩了

目錄

在這里插入圖片描述

class類

BaseApi

import type LocalDB from '@/class/DbClass'
import { tableConfig } from '@/enums/DbEnums'
import { CallBackFunType } from '@/utils/request'
class BaseApi {public localDB: LocalDB;public table: string;public hasTable: boolean;constructor(table: string, localDB: LocalDB){this.table = tablethis.hasTable = falsethis.localDB = localDBif(this.localDB.Db().isOpen()){this.localDB.Db().hasTable(table).then((res:any) => {this.hasTable = res}).catch((res: any) => {console.log("構造初始化失敗, table: " + this.table)})}}public getTable(): string{return this.table;}public getHasTable(): boolean{return this.hasTable;}public setHasTable(hasTable:boolean){this.hasTable = hasTable;}/*** 驗證是否存在表信息*/public async verifyTable(init: boolean = false): Promise<CallBackFunType<any>>{if(!this.getHasTable() && init){const res = await this.init();if(!res){return new CallBackFunType({}).fail('初始化數據庫失敗')}else {return new CallBackFunType({}).success('初始化數據庫成功');}}return new CallBackFunType({}).success('存在數據庫');}/*** 初始化*/public async init(): Promise<boolean>{// 瀏覽器不能直接新建,需要適配const res = await this.localDB.Db().createTable(this.table, tableConfig[this.table])return res;}
}export default BaseApi;

DbClass

import type SqlAbapter from '@/plugins/db/SqlAbapter'// 根據平臺選擇適配器
class LocalDB {private db : SqlAbapter;public name : string;public isSupport: boolean;constructor(db:SqlAbapter , name: string, isSupport: boolean){this.db = db;this.name = namethis.isSupport = isSupport}/*** 批量初始化表*/public initDb(tables: Record<string, Record<string, any>>): void{// 驗證當前是否開啟了離線緩存,驗證當前是否支持離線緩存if(!this.isSupport){console.log("當前不支持離線緩存")return;}this.db.open(tables).then((res: boolean) => {if(res){console.log('數據庫打開成功')}else {console.log('數據庫打開失敗')}})}// 驗證是否public Db():SqlAbapter{// 驗證當前是否開啟了離線緩存,驗證當前是否支持離線緩存if(!this.isSupport){console.log("當前不支持離線緩存")}return this.db;}
}export default LocalDB;

方法抽象

抽象模版 SqlAbapter.ts


interface SqlAbapter {/*** 數據庫是否開啟*/isOpen: () => boolean;/*** 開啟數據庫*/open: (tables: Record<string, Record<string, any>>) => Promise<boolean>;/*** 關閉數據庫*/close: () => Promise<boolean>;/*** 是否存在表*/hasTable: (dbTable: string) =>  Promise<boolean>;/*** 創建表*/createTable: (dbTable: string, data: string[]) => Promise<boolean>;/*** 刪除表*/dropTable: (dbTable: string) => Promise<boolean>;/*** 新增數據:  indexedDb 重讀id會報錯*/insertTableData: (dbTable: string, data:Record<string,any>) => Promise<boolean>;/*** 新增或修改*/insertOrReplaceData: (dbTable:string, data:Record<string,any>, condition:Record<string,any>) => Promise<boolean>;/*** 查詢數據*/selectTableData: (dbTable:string, condition: Record<string,any>) => Promise<any>;/*** 刪除數據*/deleteTableData: (dbTable:string, condition: Record<string,any>) => Promise<boolean>;/*** 更新數據*/updateTableData: (dbTable:string, data: Record<string,any>, condition: Record<string,any>) => Promise<Boolean>;/*** 分頁查詢*/pullSQL: (dbTable: string, id:string, current: number, size:number) => Promise<any>;}export default SqlAbapter;

sqlite實現

import type SqlAbapter from '@/plugins/db/SqlAbapter'
import { replace, keyValSql, isEmpty, whereSql, updateSetSql } from '@/plugins/utils';
class SqliteAdapter implements SqlAbapter {public dbName: string;public dbPath: string;constructor(dbName: string, dbPath:string){this.dbName = dbName;this.dbPath = dbPath;}// 判斷數據庫是否打開isOpen() {// 數據庫打開了就返回 true,否則返回 falsevar open = plus.sqlite.isOpenDatabase({name: this.dbName,  // 數據庫名稱path: this.dbPath  // 數據庫地址})return open;}// 創建數據庫 或 有該數據庫就打開open(tables: Record<string, Record<string, any>>):Promise<boolean>  {return new Promise((resolve, reject) => {// 打開數據庫plus.sqlite.openDatabase({name: this.dbName,path: this.dbPath,success(e) {resolve(true); // 成功回調// 初始化表 todo: },fail(e) {reject(false);  // 失敗回調}})})}hasTable(dbTable: string): Promise<boolean> {let sql = `select * from sqlite_master where type = 'table' and name = '${dbTable}'`return new Promise((resolve, reject) => {// 打開數據庫plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(res) {if (res.resultSet.length > 0) {resolve(true); // 成功回調} else {resolve(false);  // 失敗回調}},fail(e) {reject(false);  // 失敗回調}})})}// 關閉數據庫close(): Promise<boolean>  {return new Promise((resolve, reject) => {plus.sqlite.closeDatabase({name: this.dbName,success(e) {resolve(e);},fail(e) {reject(e);}})})}// 數據庫建表 sql:'CREATE TABLE IF NOT EXISTS dbTable("id" varchar(50),"name" TEXT) // 創建 CREATE TABLE IF NOT EXISTS 、 dbTable 是表名,不能用數字開頭、括號里是表格的表頭createTable(dbTable: string, data: string[]): Promise<boolean> {let keys = '';if(!data || data.length <= 0){return new Promise((resolve, reject) => { reject("創建失敗,索引不能為空") })}data.forEach((key:string) => {keys = keys + key +",";})keys = replace(keys, ",")// todo: 增加表 屬性, varchar等等let sql = `CREATE TABLE IF NOT EXISTS ${dbTable}(${keys})`return new Promise((resolve, reject) => {// executeSql: 執行增刪改等操作的SQL語句plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(e) {resolve(true);},fail(e) {reject(false);}})})}// 數據庫刪表 sql:'DROP TABLE dbTable'dropTable(dbTable: string): Promise<boolean> {return new Promise((resolve, reject) => {plus.sqlite.executeSql({name: this.dbName,sql: [`DROP TABLE ${dbTable}`],success(e) {resolve(true);},fail(e) {reject(false);}})})}// 向表格里添加數據 sql:'INSERT INTO dbTable VALUES('x','x','x')'   對應新增// 或者 sql:'INSERT INTO dbTable ('x','x','x') VALUES('x','x','x')'   具體新增// 插入 INSERT INTO  、 dbTable 是表名、根據表頭列名插入列值insertTableData(dbTable: string, data:Record<string,any>): Promise<boolean> {// 判斷有沒有傳參if (dbTable !== undefined && data) {// 判斷傳的參是否有值if (!isEmpty(data)) {let res = keyValSql(data)// 拼接sql,執行插入var sql = `INSERT INTO ${dbTable} (${res.keySql}) VALUES(${res.valSql})`;// console.log(sql);return new Promise((resolve, reject) => {// 表格添加數據plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(e) {resolve(e);},fail(e) {reject(e);}})})} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}}// 根據條件向表格里添加數據  有數據更新、無數據插入// (建表時需要設置主鍵) 例如 --- "roomid" varchar(50) PRIMARY KEYinsertOrReplaceData(dbTable: string, data: Record<string,any>):Promise<boolean> {// 判斷有沒有傳參if (dbTable !== undefined && data) {if (!isEmpty(data)) {let res = keyValSql(data)let sql = `INSERT OR REPLACE INTO ${dbTable} (${res.keySql}) VALUES(${res.valSql})`;// console.log(sql);return new Promise((resolve, reject) => {// 表格添加數據plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(e) {resolve(e);},fail(e) {reject(e);}})})}else {return new Promise((resolve, reject) => { reject("錯誤添加") })}} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}}// 查詢獲取數據庫里的數據 sql:'SELECT * FROM dbTable WHERE lname = 'lvalue''// 查詢 SELECT * FROM 、 dbTable 是表名、 WHERE 查找條件 lname,lvalue 是查詢條件的列名和列值selectTableData(dbTable:string,  condition: Record<string,any>): Promise<any> {if (dbTable !== undefined) {// 第一個是表單名稱,后兩個參數是列表名,用來檢索let where = ''if(!isEmpty(condition)){where = whereSql(condition)}// if (lname !== undefined && cc !== undefined) {// 	// 兩個檢索條件// 	var sql = `SELECT * FROM ${dbTable} WHERE ${lname} = '${lvalue}' AND ${cc} = '${dd}'`;// }let sql = `SELECT * FROM ${dbTable}`;if(where){sql = sql + " where " + where;}return new Promise((resolve, reject) => {// 表格查詢數據  執行查詢的SQL語句plus.sqlite.selectSql({name: this.dbName,sql: sql,success(e) {resolve(e);},fail(e) {reject(e);}})})} else {return new Promise((resolve, reject) => { reject("錯誤查詢") });}}// 刪除表里的數據 sql:'DELETE FROM dbTable WHERE lname = 'lvalue''// 刪除 DELETE FROM 、 dbTable 是表名、 WHERE 查找條件 lname,lvalue 是查詢條件的列名和列值deleteTableData(dbTable:string, condition: Record<string,any>): Promise<boolean> {if (dbTable !== undefined) {let where = ''if(!isEmpty(condition)){where = whereSql(condition)}var sql = `DELETE FROM ${dbTable}`;if(where){sql = sql + " where " + where}return new Promise((resolve, reject) => {// 刪除表數據plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(e) {resolve(e);},fail(e) {reject(e);}})})} else {return new Promise((resolve, reject) => { reject("錯誤刪除") });}}// 修改數據表里的數據 sql:"UPDATE dbTable SET 列名 = '列值',列名 = '列值' WHERE lname = 'lvalue'"// 修改 UPDATE 、 dbTable 是表名, data: 要修改的列名=修改后列值, lname,lvalue 是查詢條件的列名和列值updateTableData(dbTable:string, data: Record<string,any>, condition: Record<string,any>): Promise<boolean> {if(!dbTable || isEmpty(data)){return new Promise((resolve, reject) => { reject("修改錯誤") });}let res = updateSetSql(data)var sql = `UPDATE ${dbTable} SET ` + res;if(!isEmpty(condition)){let where = whereSql(condition)if(where){sql = sql + " where " + where}}// WHERE 前面是要修改的列名、列值,后面是條件的列名、列值return new Promise((resolve, reject) => {// 修改表數據plus.sqlite.executeSql({name: this.dbName,sql: [sql],success(e) {resolve(e);},fail(e) {reject(e);}})})}// 獲取指定數據條數  sql:"SELECT * FROM dbTable ORDER BY 'id' DESC LIMIT 15 OFFSET 'num'"// dbTable 表名, ORDER BY 代表排序默認正序, id 是排序的條件 DESC 代表倒序,從最后一條數據開始拿// LIMIT 15 OFFSET '${num}',這句的意思是跳過 num 條拿 15 條數據, num 為跳過多少條數據是動態值// 例 初始num設為0,就從最后的數據開始拿15條,下次不拿剛獲取的數據,所以可以讓num為15,這樣就能一步一步的拿完所有的數據pullSQL(dbTable: string, id:string, current: number, size:number): Promise<any> {if(current <= 0){return new Promise((resolve, reject) => { reject("分頁查詢錯誤,頁碼必須從1開始") });}let num = 0;if(current > 0){num = (current - 1) * size}return new Promise((resolve, reject) => {plus.sqlite.selectSql({name: this.dbName,sql: `SELECT * FROM ${dbTable} ORDER BY '${id}' DESC LIMIT ${size} OFFSET '${num}'`,success(e) {resolve(e);},fail(e) {reject(e);}})})}
}export default SqliteAdapter;

indexedDb 實現

import type SqlAbapter from '@/plugins/db/SqlAbapter'
import { keyValSql, isEmpty, whereSql, updateSetSql } from '@/plugins/utils';class IndexDbAdapter implements SqlAbapter {// 這個做法是因為 不同的瀏覽器獲取indexedDB的方式不一樣。// mozIndexedDB:火狐瀏覽器內核;webkitIndexedDB:webkit內核;msIndexedDB:IE內核。public indexedDB = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB;public dbName: string;public dbPath: string;private request: IDBOpenDBRequest | undefined;private db: IDBDatabase | undefined;constructor(dbName: string, dbPath:string){this.dbName = dbName;this.dbPath = dbPath;// this.request = this.indexedDB.open(this.dbName,1)}// 判斷數據庫是否打開isOpen() {// 數據庫打開了就返回 true,否則返回 falseif(this.request == null){return false;}else {return true;}}// 創建數據庫 或 有該數據庫就打開open(tables: Record<string, Record<string, any>>):Promise<boolean>  {return new Promise((resolve, reject) => {this.request = this.indexedDB.open(this.dbName,1)let _this = this;this.request.onerror = function(event) {console.error('Database error:', event.target?.errorCode);reject("數據庫打開失敗");};this.request.onupgradeneeded = function(event) {console.log('onupgradeneeded ====>', event)resolve(true);_this.db =  (event.target as IDBOpenDBRequest).result;// _this.createTable('ttt', ['name', 'val'])if(!isEmpty(tables)){let tabs = Object.keys(tables)tabs.forEach((tab:string) => {_this.hasTable(tab).then((res: boolean) => {if(res){// 當前console.log(`當前數據庫: ${tab} 已存在,不進行初始化`)}else {// 初始化表_this.createTable(tab, Object.keys(tables[tab])).then((creRes:any) => {if(creRes){console.log(`當前數據庫: ${tab} 初始化完成`)}else {console.log(`當前數據庫: ${tab} 初始化失敗`)}})}})})}};this.request.onsuccess = function(event) {_this.db = (event.target as IDBOpenDBRequest).result;resolve(true);console.log('Database opened successfully');};})}hasTable(dbTable: string): Promise<boolean> {let _this = this;return new Promise((resolve, reject) => {let res = _this.db?.objectStoreNames.contains(dbTable) as booleanconsole.log('hasTable this.db? =====>', this.db)console.log('hasTable =====>', res)resolve(res);})}// 關閉數據庫close(): Promise<boolean>  {let _this = this;return new Promise((resolve, reject) => {_this.db?.close();resolve(true);})}// 數據庫建表 sql:'CREATE TABLE IF NOT EXISTS dbTable("id" varchar(50),"name" TEXT) // 創建 CREATE TABLE IF NOT EXISTS 、 dbTable 是表名,不能用數字開頭、括號里是表格的表頭createTable(dbTable: string, data: string[]): Promise<boolean> {if(!data || data.length <= 0){return new Promise((resolve, reject) => { reject("創建失敗,索引不能為空") })}let _this = this;return new Promise((resolve, reject) => {// executeSql: 創建表const objectStore = _this.db?.createObjectStore(dbTable,{ keyPath: 'id' });data.forEach((key:any) => {objectStore?.createIndex(key, key,  {unique: false})})resolve(true)})}// 數據庫刪表 sql:'DROP TABLE dbTable'dropTable(dbTable: string): Promise<boolean> {return new Promise((resolve, reject) => {reject(false);})}// 向表格里添加數據 sql:'INSERT INTO dbTable VALUES('x','x','x')'   對應新增// 或者 sql:'INSERT INTO dbTable ('x','x','x') VALUES('x','x','x')'   具體新增// 插入 INSERT INTO  、 dbTable 是表名、根據表頭列名插入列值insertTableData(dbTable: string, data:Record<string,any>): Promise<boolean> {// 判斷有沒有傳參if (dbTable !== undefined && data) {// 判斷傳的參是否有值if (!isEmpty(data)) {let _this = this;return new Promise((resolve, reject) => {// 添加數據到對象存儲空間let transaction = _this.db?.transaction([dbTable], 'readwrite');const objectStore = transaction?.objectStore(dbTable);let request = objectStore?.add(data) as IDBRequest;// 寫入數據的事件監聽request.onsuccess = function (event) {resolve(true)console.log('數據寫入成功');};request.onerror = function (event) {reject("數據寫入失敗: " + event?.target?.error?.message)console.log('數據寫入失敗: event =====》 ', event);}})} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}}// 根據條件向表格里添加數據  有數據更新、無數據插入// (建表時需要設置主鍵) 例如 --- "roomid" varchar(50) PRIMARY KEYinsertOrReplaceData(dbTable: string, data: Record<string,any>):Promise<boolean> {// 判斷有沒有傳參if (dbTable !== undefined && data) {if (!isEmpty(data)) {return new Promise((resolve, reject) => {// 表格添加數據reject(false);})}else {return new Promise((resolve, reject) => { reject("錯誤添加") })}} else {return new Promise((resolve, reject) => { reject("錯誤添加") })}}// 查詢獲取數據庫里的數據 sql:'SELECT * FROM dbTable WHERE lname = 'lvalue''// 查詢 SELECT * FROM 、 dbTable 是表名、 WHERE 查找條件 lname,lvalue 是查詢條件的列名和列值selectTableData(dbTable:string,  condition: Record<string,any>): Promise<any> {if (dbTable !== undefined) {// 第一個是表單名稱,后兩個參數是列表名,用來檢索return new Promise((resolve, reject) => {reject(false);})} else {return new Promise((resolve, reject) => { reject("錯誤查詢") });}}// 刪除表里的數據 sql:'DELETE FROM dbTable WHERE lname = 'lvalue''// 刪除 DELETE FROM 、 dbTable 是表名、 WHERE 查找條件 lname,lvalue 是查詢條件的列名和列值deleteTableData(dbTable:string, condition: Record<string,any>): Promise<boolean> {if (dbTable !== undefined) {return new Promise((resolve, reject) => {reject(false);})} else {return new Promise((resolve, reject) => { reject("錯誤刪除") });}}// 修改數據表里的數據 sql:"UPDATE dbTable SET 列名 = '列值',列名 = '列值' WHERE lname = 'lvalue'"// 修改 UPDATE 、 dbTable 是表名, data: 要修改的列名=修改后列值, lname,lvalue 是查詢條件的列名和列值updateTableData(dbTable:string, data: Record<string,any>, condition: Record<string,any>): Promise<boolean> {return new Promise((resolve, reject) => {reject(false);})}// 獲取指定數據條數  sql:"SELECT * FROM dbTable ORDER BY 'id' DESC LIMIT 15 OFFSET 'num'"// dbTable 表名, ORDER BY 代表排序默認正序, id 是排序的條件 DESC 代表倒序,從最后一條數據開始拿// LIMIT 15 OFFSET '${num}',這句的意思是跳過 num 條拿 15 條數據, num 為跳過多少條數據是動態值// 例 初始num設為0,就從最后的數據開始拿15條,下次不拿剛獲取的數據,所以可以讓num為15,這樣就能一步一步的拿完所有的數據pullSQL(dbTable: string, id:string, current: number, size:number): Promise<any> {return new Promise((resolve, reject) => {reject(false);})}
}export default IndexDbAdapter;

動態選取

db-plugins

import SqliteAdapter from './db/sqlite/sqliteAdapter';
import IndexDbAdapter from './db/indexedDb/indexedDbAdapter';
import { DbConfig, tableConfig } from '@/enums/DbEnums'
import  LocalDB from '@/class/DbClass'let localDB : LocalDB;
// #ifdef APP-PLUS
//  App 環境使用 sqlite 適配器localDB = new LocalDB(new SqliteAdapter(DbConfig.Name, DbConfig.Path), 'sqlite', true);
// #endif
// #ifdef  MP-WEIXIN
// 小程序環境使用內存適配器,自定義實現// #endif
// #ifdef H5
//  瀏覽器 環境使用 indexeddb 適配器localDB = new LocalDB(new IndexDbAdapter(DbConfig.Name, DbConfig.Path), 'indexedDb', true);
// #endifconsole.log('當前環境注冊的本地數據庫為',localDB.name)// 初始化表
localDB.initDb(tableConfig)
export default localDB;

api應用

testService

import {request , CallBackFunType} from '@/utils/request'
import { HttpPath } from '@/enums/constant'
import localDB from '@/plugins/db-plugins'
import type LocalDB from '@/class/DbClass'
import { DbTable, DbConfig } from '@/enums/DbEnums'
import BaseApi from '@/class/baseApiClass'class TestApi extends BaseApi{constructor(table: string, localDB:LocalDB){super(table, localDB)}public save(data: {id: string ,name: string, value: string}) {return request({'url': HttpPath.App + '/dict/type','method': 'post',data}, async (requestConfig: any) => {console.log('離線回調操作: ', requestConfig)const res = await this.verifyTable(true);return new Promise<CallBackFunType<any>>((resolve, reject) => {if(res.code != 200){reject(res)}else{// 執行數據庫操作this.localDB.Db().insertTableData(DbTable.Test, data).then((res) => {if(res){resolve(new CallBackFunType({}).success())}else {reject(new CallBackFunType({}).fail())}}).catch((e:any) => {reject(e)})}})})}
}export default new TestApi(DbTable.Test, localDB);

頁面使用

<template><view>測試</view>
</template><script lang="ts" setup>import { onLoad } from '@dcloudio/uni-app';import TestApi from '@/api/testService'onLoad(() => {TestApi.save({id:'1231',name: '2321', value: 'dasdas'}).then((res:any) => {console.log(res)})})
</script><style lang="scss">
</style>

執行效果 以瀏覽器(indexedDb)為例

在這里插入圖片描述
在這里插入圖片描述
在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/70747.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/70747.shtml
英文地址,請注明出處:http://en.pswp.cn/web/70747.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

HIVE SQL函數之比較函數

背景&#xff1a;今天接到一個臨時需求&#xff0c;需要比較abc的大小&#xff0c;但是abc三個字段都存在為空的情況。 開發&#xff1a;對于這個開發很簡單&#xff0c;因為比較函數有太多了&#xff0c;首先想到的是用case when去進行一個非空的判斷&#xff0c;再去比較用I…

AI探索筆記:淺談人工智能算法分類

人工智能算法分類 這是一張經典的圖片&#xff0c;基本概況了人工智能算法的現狀。這張圖片通過三個同心圓展示了人工智能、機器學習和深度學習之間的包含關系&#xff0c;其中人工智能是最廣泛的范疇&#xff0c;機器學習是其子集&#xff0c;專注于數據驅動的算法改進&#…

進程概念、PCB及進程查看

文章目錄 一.進程的概念進程控制塊&#xff08;PCB&#xff09; 二.進程查看通過指令查看進程通過proc目錄查看進程的cwd和exe獲取進程pid和ppid通過fork()創建子進程 一.進程的概念 進程是一個運行起來的程序&#xff0c;而程序是存放在磁盤的&#xff0c;cpu要想執行程序的指…

OA辦公系統自動滲透測試過程

目錄 一、下載環境源碼 二、部署環境 三、測試 XSS漏洞 SQL注入 文件上傳漏洞 一、下載環境源碼 OA源碼打包地址: https://download.csdn.net/download/weixin_43650289/90434502?spm=1001.2014.3001.5503 二、部署環境

怎么修改node_modules里的文件,怎么使用patch-package修改node_modules的文件,怎么修改第三方庫原文件。

在開發中會遇到需要node_modules里第三方庫有bug&#xff0c;然后需要修改node_modules文件的情況 使用patch-package包可以修改node_modules里的文件 patch-package npm 官網&#xff1a;patch-package - npm 安裝 npm i patch-package 修改文件后 npx patch-package s…

Python在實際工作中的運用-通用格式CSV文件自動轉換XLSX

繼續上篇《Python在實際工作中的運用-CSV無損轉XLSX的幾個方法》我們雖然對特定格式的CSV實現了快速轉換XLSX的目標,但是在運行Py腳本前,還是需要編輯表格創建腳本和數據插入腳本,自動化程度很低,實用性不強,為減少人工提高效率,實現輸入CSV文件路徑即可自動適配完成轉換…

seacmsv9報錯注入

1、seacms的介紹 ? seacms中文名&#xff1a;海洋影視管理系統。是一個采用了php5mysql架構的影視網站框架&#xff0c;因此&#xff0c;如果該框架有漏洞&#xff0c;那使用了該框架的各個網站都會有相同問題。 2、源碼的分析 漏洞的部分源碼如下&#xff1a; <?php …

Hbase客戶端API——語句大全

目錄 創建表&#xff1a; 插入數據&#xff1a; 刪除數據&#xff1a; 修改數據&#xff1a; 查詢數據&#xff1a;Get 查詢數據&#xff1a;Scan 查詢數據&#xff1a;過濾查詢 創建表&#xff1a; 檢驗&#xff1a; 插入數據&#xff1a; 驗證 一次多條數據插入 驗證&…

vscode 版本

vscode官網 Visual Studio Code - Code Editing. Redefined 但是官網只提供最新 在之前的版本就要去github找了 https://github.com/microsoft/vscode/releases 獲取舊版本vscode安裝包的方法_vscode 老版本-CSDN博客

IP------PPP協議

這只是IP的其中一塊內容PPP&#xff0c;IP還有更多內容可以查看IP專欄&#xff0c;前一章內容為網絡類型&#xff0c;可通過以下路徑查看IP---網絡類型-CSDN博客&#xff0c;歡迎指正 3.PPP協議 1.PPP優點 網絡類型&#xff1a;p2p PPP---點到點協議 兼容性會更強凡是接口或…

Springboot基礎篇(3):Bean管理

前言&#xff1a;Spring 通過掃描類路徑&#xff08;Classpath&#xff09;來查找帶有特定注解&#xff08;如 Component、Service、Repository 等&#xff09;的類&#xff0c;并將它們注冊為 Spring 容器中的 Bean。 1 Bean掃描 Bean 掃描是 Spring 框架的核心功能之一&…

Metal 學習筆記二:3D模型

是什么讓一個好游戲更好玩&#xff1f;漂亮的圖像&#xff01;就像《神界&#xff1a;原罪2》&#xff0c;《暗黑破壞神3》以及《巫師3》等大作一樣&#xff0c;需要一個強大的程序團隊以及3D美術團隊強強合作。你在屏幕中看到正是3D模型使用自定義渲染繪制的結果。就像上一章你…

【算法】797. 差分

題目 797. 差分 思路 差分的實質是通過構造數組b減少時間復雜度&#xff0c;數組a為初始數據&#xff0c;構造數組b&#xff0c;數組a是b的前綴和&#xff0c;通過對數組b操作就可以實現數組a每個數加上c&#xff0c;而對數組b的操作在單位時間內即可完成&#xff0c;對數組…

解鎖狀態模式:Java 編程中的行為魔法

系列文章目錄 后續補充~~~ 文章目錄 一、狀態模式&#xff1a;概念與原理二、狀態模式的深度剖析&#xff08;一&#xff09;模式定義與核心思想&#xff08;二&#xff09;模式結構與角色 三、狀態模式的實際應用場景&#xff08;一&#xff09;電商系統中的訂單狀態管理&…

php 獲取head參數

php 獲取head參數 在PHP中&#xff0c;獲取HTTP頭部&#xff08;head&#xff09;參數可以通過不同的方式實現&#xff0c;下面為你詳細介紹幾種常見的方法。 1. 使用$_SERVER超全局變量 $_SERVER 是PHP中的一個超全局變量&#xff0c;它包含了諸如頭信息、路徑、腳本位置等…

數據結構與算法-圖論-最短路-拓展運用

選擇最佳路線 分析&#xff1a; 這是一道圖論中的最短路徑問題&#xff0c;目標是在給定的公交網絡中&#xff0c;找到從琪琪家附近的車站出發&#xff0c;到她朋友家附近車站&#xff08;編號為 s &#xff09;的最短時間。以下是對該問題的詳細分析&#xff1a; 問題關鍵信息…

AI知識架構之神經網絡

神經網絡:這是整個內容的主題,是一種模擬人類大腦神經元結構和功能的計算模型,在人工智能領域廣泛應用。基本概念:介紹神經網絡相關的基礎概念,為后續深入理解神經網絡做鋪墊。定義與起源: 神經網絡是模擬人類大腦神經元結構和功能的計算模型,其起源于對生物神經系統的研…

【江科協-STM32】5. 輸出比較

1. 輸出比較簡介 OC(Output Compare)輸出比較。 輸出比較可以通過CNT&#xff08;CNT計數器&#xff09;與CCR寄存器值的關系&#xff0c;來對輸出電平進行置1、置0或翻轉的操作&#xff0c;用于輸出一定頻率和占空比的PWM波形。 :::tip CNT計數器是正向計數器。它只能正向累…

C++ Primer 再探迭代器

歡迎閱讀我的 【CPrimer】專欄 專欄簡介&#xff1a;本專欄主要面向C初學者&#xff0c;解釋C的一些基本概念和基礎語言特性&#xff0c;涉及C標準庫的用法&#xff0c;面向對象特性&#xff0c;泛型特性高級用法。通過使用標準庫中定義的抽象設施&#xff0c;使你更加適應高級…

排查和解決線程池瓶頸問題案例

在分布式系統中&#xff0c;線程池的使用非常普遍&#xff0c;尤其是在處理異步任務時。然而&#xff0c;線程池的配置不當可能會導致性能瓶頸&#xff0c;進而影響系統的整體性能。本文將分享一個實際案例&#xff0c;介紹如何通過日志分析和線程池優化來解決系統中的性能瓶頸…