网站首页 > 技术文章 正文
大家好,很高兴又见面了,我是"高级前端?进阶?",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发,您的支持是我不断创作的动力。
1.什么是 sql.js
sql.js is a port of SQLite to Webassembly, by compiling the SQLite C code with Emscripten. It uses a virtual database file stored in memory, and thus doesn't persist the changes made to the database. However, it allows you to import any existing sqlite file, and to export the created database as a JavaScript typed array.
sql.js 是 SQLite 到 WebAssembly 的端口,通过使用 Emscripten 编译 SQLite C 代码。
sql.js 允许开发者创建关系数据库并完全在浏览器中查询,而且使用存储在内存中的虚拟数据库文件,因此不会保留对数据库所做的更改。 但是,其允许开发者导入任何现有的 sqlite 文件,并将创建的数据库导出为 JavaScript 类型数组。
sql.js 可以像任何传统的 JavaScript 库一样使用,因此如果正在使用 JavaScript 构建本机应用程序(例如使用 Electron),或者正在使用 Node.js 则可能更喜欢使用 SQLite 到 JavaScript 的本机绑定 (Native Binding of SQLite to JavaScript)。
本机绑定不仅速度更快,而且还能够直接处理数据库文件,而不必将整个数据库加载到内存中,从而避免内存不足错误并进一步提高性能。
目前 sql.js 在 Github 通过 MIT 协议开源,有超过 12.2k 的star、1k 的fork、项目依赖量6.8k、代码贡献者 50+ ,妥妥的前端优质开源项目。
2.如何使用 sql.js
浏览器中使用
默认情况下,sql.js 使用 wasm,因此除了 javascript 库之外还需要加载 .wasm 文件。 从 npm 安装 sql.js 后,开发者可以在 ./node_modules/sql.js/dist/sql-wasm.wasm 中找到此文件,并指示捆绑程序将其添加到静态资产或从 CDN 加载。
然后使用传递给 initSqlJs 的配置对象的 locateFile 属性来指示文件所在的位置。 如果使用 webpack 等构建器,则可以自动执行此操作。
const initSqlJs = require('sql.js');
const SQL = await initSqlJs({
// 需要异步加载 wasm 二进制文件
// 在 node 中运行时可以完全省略 locateFile
locateFile: file => `https://sql.js.org/dist/${file}`
});
// 创建数据库
const db = new SQL.Database();
// 执行包含多个语句的单个 SQL 字符串
let sqlstr = "CREATE TABLE hello (a int, b char); \
INSERT INTO hello VALUES (0, 'hello'); \
INSERT INTO hello VALUES (1, 'world');";
db.run(sqlstr);
// 准备一条 sql 语句
const stmt = db.prepare("SELECT * FROM hello WHERE a=:aval AND b=:bval");
// 将值绑定到参数并获取查询结果
const result = stmt.getAsObject({':aval' : 1, ':bval' : 'world'});
console.log(result);
// 输出值 {a:1, b:'world'}
stmt.bind([0, 'hello']);
stmt.free();
const res = db.exec("SELECT * FROM hello");
/* 输出
[
{columns:['a','b'], values:[[0,'hello'],[1,'world']]}
]
*/
db.create_aggregate(
"json_agg",
{
init: () => [],
step: (state, val) => [...state, val],
finalize: (state) => JSON.stringify(state),
}
);
db.exec("SELECT json_agg(column1) FROM (VALUES ('hello'), ('world'))");
// -> 输出值 '["hello","world"]'
// 将数据库导出到包含 SQLite 数据库文件的 Uint8Array
const binaryArray = db.export();
Node.js 环境中
sql.js 托管在 npm 上,只需运行 npm install sql.js 即可。或者,开发者可以简单地从下面的下载链接下载 sql-wasm.js 和 sql-wasm.wasm。
在 Node.js 环境中,可以通过下面代码从磁盘读取数据库:
const fs = require('fs');
const initSqlJs = require('sql-wasm.js');
const filebuffer = fs.readFileSync('test.sqlite');
initSqlJs().then(function(SQL){
// 读取 DB
const db = new SQL.Database(filebuffer);
});
同时,可以通过下面方法将内容写入磁盘:
const fs = require("fs");
// [...] (create the database)
const data = db.export();
const buffer = Buffer.from(data);
fs.writeFileSync("filename.sqlite", buffer);
同时,如果开发者不想在主应用程序线程中运行 CPU 密集型 SQL 查询,则可以使用更受限制的 WebWorker API。
const worker = new Worker("/dist/worker.sql-wasm.js");
worker.onmessage = () => {
console.log("Database opened");
worker.onmessage = event => {
console.log(event.data);
// 查询结果
};
worker.postMessage({
id: 2,
action: "exec",
sql: "SELECT age,name FROM test WHERE id=$id",
params: {"$id": 1}
});
};
worker.onerror = e => console.log("Worker error:", e);
worker.postMessage({
id:1,
action:"open",
buffer:buf,
/* 可选,代表 SQLite 数据库文件的 ArrayBuffer*/
});
sql.js 还支持与 XMLHttpRequest 一起使用,比如下面的示例:
const xhr = new XMLHttpRequest();
// For example: https://github.com/lerocha/chinook-database/raw/master/ChinookDatabase/DataSources/Chinook_Sqlite.sqlite
xhr.open('GET', '/path/to/database.sqlite', true);
xhr.responseType = 'arraybuffer';
xhr.onload = e => {
const uInt8Array = new Uint8Array(xhr.response);
const db = new SQL.Database(uInt8Array);
const contents = db.exec("SELECT * FROM my_table");
// contents is now [{columns:['col1','col2',...], values:[[first row], [second row], ...]}]
};
xhr.send();
更多关于 Sql.js 的知识和用法可以参考文末资料,本文不再过多展开。
参考资料
https://madewithwebassembly.com/showcase/sqlite/
https://github.com/sql-js/sql.js
https://www.youtube.com/watch?app=desktop&v=0DZ472GiVNw
https://blog.logrocket.com/detailed-look-basic-sqljs-features/
猜你喜欢
- 2025-07-28 JDK高版本特性总结与ZGC实践(jdk高版本兼容低版本吗)
- 2025-07-28 分布式事务解决方案之2PC(两阶段提交)
- 2025-07-28 Oracle 赢得云大单:一年 2100 亿元
- 2025-07-28 Innodb中的事务隔离级别和锁的关系
- 2025-07-28 服务器密码错误被锁定怎么解决(服务器密码忘了怎么解除)
- 2025-07-28 2分钟,快速认识什么是SQL(什么是sql?)
- 2025-07-28 哪些百分百会问的MySQ面试题(面试mysql问题)
- 2025-07-28 在实际操作过程中如何避免出现SQL注入漏洞
- 2025-07-28 软件项目中的数据库设计规范模板(软件项目中的数据库设计规范模板有哪些)
- 2025-07-28 使用DDL创建数据库&数据表时需要注意什么?
你 发表评论:
欢迎- 641℃几个Oracle空值处理函数 oracle处理null值的函数
- 631℃Oracle分析函数之Lag和Lead()使用
- 623℃0497-如何将Kerberos的CDH6.1从Oracle JDK 1.8迁移至OpenJDK 1.8
- 619℃Oracle数据库的单、多行函数 oracle执行多个sql语句
- 615℃Oracle 12c PDB迁移(一) oracle迁移到oceanbase
- 605℃【数据统计分析】详解Oracle分组函数之CUBE
- 596℃最佳实践 | 提效 47 倍,制造业生产 Oracle 迁移替换
- 580℃Oracle有哪些常见的函数? oracle中常用的函数
- 最近发表
- 标签列表
-
- 前端设计模式 (75)
- 前端性能优化 (51)
- 前端模板 (66)
- 前端跨域 (52)
- 前端缓存 (63)
- 前端aes加密 (58)
- 前端脚手架 (56)
- 前端md5加密 (54)
- 前端路由 (61)
- 前端数组 (73)
- 前端js面试题 (50)
- 前端定时器 (59)
- 前端获取当前时间 (50)
- Oracle RAC (76)
- oracle恢复 (77)
- oracle 删除表 (52)
- oracle 用户名 (80)
- oracle 工具 (55)
- oracle 内存 (55)
- oracle 导出表 (62)
- oracle约束 (54)
- oracle 中文 (51)
- oracle链接 (54)
- oracle的函数 (58)
- 前端调试 (52)
本文暂时没有评论,来添加一个吧(●'◡'●)