HTML5的IndexedDB数据库

简介: HTML5的IndexedDB数据库

一、IndexedDB数据库概述

1.NoSQL数据库

新一代数据库,其含义是Not Only SQL 或non-relational,具有非关系型、高效的特点。与关系型数据库比较,NoSQL数据库适用于数据模型比较简单、高并发读写、海量数据的高效存储和访问等需求。

2.IndexedDB 数据库

(1) IndexedDB 数据库和对象仓库

一个网站可能有一个或多个IndexedDB 数据库,每个数据库必须具有惟一的名称;一个数据库可包含一个或多个对象仓库。

(2) 版本更新和事务处理

IndexedDB 数据库中创建或删除对象仓库、创建或删除索引的操作,可以看作是数据库的结构发生变化,必须使用新的版本号来更新数据库的版本,以避免重复修改数据库结构。

(3) 索引和游标

IndexedDB数据库中,只能对被索引的属性值进行检索。对象仓库可有一个或多个索引。

IndexedDB 中的游标能够迭代一个对象仓库中的所有记录。IndexedDB 中的游标是双向的,所以可以向前和向后迭代记录,还可以跳过非惟一索引中的重复记录。

 

3.IndexedDB的异步API   

IndexedDB规范中包含异步 API 和同步 API。同步API 用于Web浏览器中。IndexedDB大部分操作的结果返回模式,使用异步API的请求—响应模式,所有异步请求都有一个onsuccess回调函数和一个onerror回调函数,前者在数据库操作成功时调用,后者在一个操作未成功时调用。

 

二、创建数据库

1.创建和连接数据库

IndexedDB API中的window.indexedDB的open()方法,用于创建或打开指定的数据库,语法格式如下。

    var dbRequest = indexedDB.open(dbName,dbVersion);

 
QQ截图20220204224139.png

 

2. IndexedDB 数据库中的对象定义

    IndexedDB 数据库包括indexedDB、IDBTransaction 、IDBKeyRange、IDBCursor等对象,为了确保脚本代码在各浏览器中兼容,应当在使用数据库之前统一定义,代码如下。

window.indexedDB=window.indexedDB||window.webkitIndexedDB;

window.IDBTransaction=window.IDBTransaction||window.webkitIDBTransaction|;

window.IDBKeyRange= window.IDBKeyRange||window.webkitIDBKeyRange;

window.IDBCursor= window.IDBCursor||window.webkitIDBCursor;

 

3. 删除数据库

  删除现有数据库,可以调用 deleteDatabase()方法,该方法的使用和open()方法类似,将要删除的数据库名称作为参数,代码如下。

function deleteDatabase() {
     var deleteDbRequest =  
             window.indexedDB.deleteDatabase(dbName);
    deleteDbRequest.onsuccess = function (e) {
      //成功处理
   };
   deleteDbRequest.onerror = function (e) {
      //错误处理
   };
}

4. 连接数据库的完整示例

代码

    <!--demo1101.html-->
    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    function connectDb() {
    var dbName = "myDb1";
    var dbVersion = 3;
    var idb;
    var dbRequest = indexedDB.open(dbName, dbVersion);
    dbRequest.onsuccess = function (e) { //连接成功的回调函数
    idb = e.target.result;
    console.log("success");
    }
    dbRequest.onerror = function (e) { //连接失败的回调函数
    console.log("error");
    }
    }
    </script>
    </head>
    <body>
    <input type="button" value="Connect" onclick="connectDb()"/>
    </body>
    </html>
    

效果

QQ截图20220204224613.png
 

三、数据库的版本更新和事务处理

1. 版本更新

连接成功的数据库还不能执行任何数据操作。在使用IndexedDB 数据库的时候,所有数据的操作都需要在一个事务内部执行。而对于对象仓库与索引的操作,类似于关系数据库中表结构或索引的操作,必须在版本更新事务内部进行。

示例实现的是一个版本更新的测试功能。

代码

    <!--demo1102.html-->
    <!DOCTYPE html>
    <html>
    <head lang="en">
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    function versionUpdate() {
    var dbName = "myDb1";
    var dbVersion = 4;
    var idb;
    var dbRequest = indexedDB.open(dbName, dbVersion);
    dbRequest.onsuccess = function (e) {
    idb = e.target.result;
    console.log("success");
    }
    dbRequest.onerror = function (e) {
    console.log("Connected error");
    }
    dbRequest.onupgradeneeded = function (e) {
    idb = e.target.result;
    var tx = e.target.transaction; //启动事物
    var oldVersion = e.oldVersion; //event的oldVersion属性
    var newVersion = e.newVersion;
    console.log("OLD :" + e.oldVersion + " NEW:" + e.newVersion);
    }
    }
    </script>
    </head>
    <body>
    <input type="button" value="Version Update" onclick="versionUpdate()"/>
    </body>
    </html>

效果

QQ截图20220204225157.png

 

2.事务处理

创建对象仓库与索引、对象仓库执行所有读取和写入操作的操作必须在事务中进行。事务具有三种模式。

  readonly:提供对某个对象仓库的只读访问,在查询对象仓库时使用。

  readwrite:提供对某个对象仓库的读取和写入权限。

  versionchange:提供读取和写入权限来修改对象仓库定义,或者创建一个新的对象仓库。

数据库的事务处理使用transaction()方法。

var tx = idb. transaction (storeNames,mode);

QQ截图20220204225348.png

四、创建对象仓库

对象仓库是数据记录的集合。在数据库中创建一个新对象仓库,需要对现有数据库进行版本更新。

要创建对象仓库,需要在数据库对象上调用 createObjectStore() 方法。

var objectStore=idb. createObjectStore(name,optionalParameters);

name为对象仓库名; optionalParameters为可选参数,是一个JSON对象(他的keyPath属性指定对象仓库中记录的主键值)

 
示例创建了名为students的对象仓库

 

QQ截图20220204225631.png

QQ截图20220204225643.png

 

五、创建索引

可以使用对象仓库的键来检索对象存储中的记录,也可以使用索引的字段来检索记录。在IndexedDB数据库中,只能对被索引的属性值进行检索。对象仓库可具有一个或多个索引。

创建索引的语法格式如下。

var idx=store.createIndex( indexName,"indexItem",optionalParameters);

其中,indexName参数值为一个字符串,表示索引名;

indexItem表示用记录对象的哪个属性来创建索引;   optionalParameters为一个JSON对象(该对象的unique属性值为true,表示数据记录的索引属性值必须是唯一的)。

 
示例创建了3个索引,其中sName是惟一索引。

    <!--demo1104.html-->
    <!DOCTYPE html>
    <html>
    <head lang="en">
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    var myDb = { //数据库名和版本号封装在JSON对象中
    dbName: "myDb1",
    dbVersion: 7 //版本更新为7
    };
    var idb;
    function createUserIndex() {
    var dbRequest = indexedDB.open(myDb.dbName, myDb.dbVersion);
    dbRequest.onsuccess = function (e) {
    console.log("Connect success");
    }
    dbRequest.onerror = function (e) {
    console.log("Connected error");
    }
    dbRequest.onupgradeneeded = function (e) {
    idb = e.target.result;
    var tx = e.target.transaction; //开启事务
    var store = tx.objectStore("students"); //获取对象仓库
    var indexName = "sName"; //索引名
    var options = {
    unique: true
    };
    var idx1 = store.createIndex(indexName, "srName", options);
    var idx2 = store.createIndex("ageIndex", "age", {unique: false});
    var idx3 = store.createIndex("addressIndex", "address", {unique: false});
    }
    }
    </script>
    </head>
    <body>
    <input type="button" value="Create Index" onclick="createUserIndex()"/>
    </body>
    </html>

QQ截图20220204225951.png

 

六、保存和删除数据

1. 保存数据

向对象仓库中插入数据,首先要连接IndexedDB 数据库。此时,open()方法中的数据库版本号参数可以省略不写。连接成功后使用该数据库对象的transaction方法开启一个读写事务,再使用put()方法或add()方法插入数据。
  

(1) put()方法或add()方法

向对象仓库添加数据可以使用put()方法或add()方法,语法格式如下。

var req = store.put(value); 

(2)插入数据的过程

向对象仓库中插入数据时,首先连接数据库,接着开启事务,获取对象仓库,最后插入数据。

开启事务后,使用transaction()方法开启事务,该方法的参数是事务的作用范围。再使用事务对象的objectStore()方法获取该事务对象的作用范围中的某个对象仓库,代码如下:

var store = tx.objectStore('students'); 

QQ截图20220204230046.png

 
示例在students对象仓库中保存了4条记录。

    <!--demo1105.html-->
    <!DOCTYPE html>
    <html>
    <head lang="en">
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    var ss = [{sId: 2, sName: "Rose", age: 22, address: "DL"},
    {sId: 5, sName: "Kellen", age: 18, address: "PK"},
    {sId: 4, sName: "Mike", age: 20, address: "SH"},
    {sId: 7, sName: "Tom", age: 18, address: "SH"},
    {sId: 6, sName: "Mary", age: 18, address: "PK"}
    ];
    var myDb = { //数据库名和版本号封装在JSON对象中
    dbName: "myDb1",
    dbVersion: 7 //版本更新为7
    };
    function inputData() {
    var idb;
    var dbRequest = indexedDB.open(myDb.dbName, myDb.dbVersion);
    dbRequest.onsuccess = function (e) {
    console.log("Connect success");
    idb = e.target.result;
    var tx = idb.transaction(["students"], "readwrite"); //开启事务
    var store = tx.objectStore("students"); //获取对象仓库
    for (var i = 0; i < ss.length; i++) {
    var req = store.put(ss[i]); //插入数据
    }
    req.onsuccess = function () {
    console.log("data input success");
    }
    req.onerror = function () {
    console.log("data input success");
    }
    }
    dbRequest.onerror = function (e) {
    console.log("Connected error");
    }
    //插入数据不需要版本更新,下面代码可略。
    dbRequest.onupgradeneeded = function (e) {
    }
    }
    </script>
    </head>
    <body>
    <input type="button" value="Insert Data" onclick="inputData()"/>
    </body>
    </html>

QQ截图20220204230133.png

 
 

2. 检索数据

(1)使用对象仓库的主键获取记录

在获取对象仓库成功后,使用对象仓库的get()方法从对象仓库中获取一条数据。

var req = store.get(key); 

该请求发出后将被立即异步执行,用户可以通过监听请求对象的onsuccess事件与请求对象的onerror事件并指定事件处理函数来定义请求被执行成功或失败时所要进行的处理.

req.onsuccess = function (e) { 

  //获取数据成功时所执行的处理 

}  

req.onerror = function(e) { 

//获取数据失败时所执行的处理 

}

 

    <!--demo1106.html-->
    <!DOCTYPE html>
    <html>
    <head lang="en">
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    function getDataByPrimarykey() {
    var dbName = "myDb1";
    var dbVersion = 7;
    var idb;
    var dbRequest = indexedDB.open(dbName, dbVersion);
    dbRequest.onsuccess = function (e) {
    console.log("Connect success");
    idb = e.target.result;
    var tx = idb.transaction(["students"], "readonly");
    var store = tx.objectStore("students");
    var req = store.get(5);
    req.onsuccess = function () {
    if (this.result == undefined) {
    alert("not found");
    } else {
    //alert(this.result.sName+" "+this.result.age);
    document.getElementById("result").innerHTML = this.result.sName + " " + this.result.age;
    }
    console.log("data input success");
    }
    req.onerror = function () {
    console.log("get Data error");
    }
    }
    dbRequest.onerror = function (e) {
    console.log("Connected error");
    }
    }
    </script>
    </head>
    <body>
    <input type="button" value="Search Data" onclick="getDataByPrimarykey()"/>
    <hr/>
    <div id="result"></div>
    </body>
    </html>

 
QQ截图20220204230322.png

 

(2)使用索引对象获取记录

除了使用主键获取记录外,如果对象仓库中存在其他属性组成的索引,则可以根据该索引的属性取得数据。

var tx = idb.transaction(['students'], "readonly"); 

var store = tx.objectStore('students'); 

var idx = store.index('sName'); 

var req = idx.get('Mike'); 

req.onsuccess = function (e) { 

//数据获取成功 

}

req.onerror = function (e) { 

//数据获取失败

}

 
QQ截图20220204230521.png

 
 

3. 删除数据

使用delete()方法可以删除对象仓库中的数据.

var req=store.delete (keyPath);

 

QQ截图20220204230611.png
 

七、使用游标检索批量数据

1. openCursor()方法及其参数

openCursor()方法用于在对象仓库中打开游标。

var req=store.openCursor(range,direction);

range参数是一个IDBKeyRange对象,该对象的返回游标的作用范围;

direction参数用于指定游标的读取方向,参数值为一个在 IndexedDB  API中预定义的常量值。

 
(1)指定游标范围的方法

创建IDBKeyRange对象的方法

QQ截图20220204230946.png

  

(2)指定游标的顺序

openCursor()方法的第2个参数direction用于指明游标的方向,有4种取值。

  IDBCursor.NEXT:顺序循环

  IDBCursor.NEXT_NO_DUPLICATE:顺序循环且键值不重复

  IDBCursor.PREV:倒序循环

  IDBCursor.PREV_NO_DUPLICATE:倒序循环且键值不重复

 

2.数据遍历

A.遍历对象仓库中的所有数据

示例11-8是遍历students对象仓库的所有记录,结合这个示例,说明遍历对象仓库中记录的步骤。

(1)连接数据库。

   init()方法完成连接数据库操作,并得到一个数据库对象idb。

   单击网页页面中的“Travel Data”按钮,显示数据的遍历结果。

 

(2)启动事务,并指定事务的作用范围。

  var tx=idb.transaction(["students"],"readonly");

  var store=tx.objectStore("students");

 

(3)打开游标。

    使用store.openCursor()方法打开游标。打开游标后,游标指向对象仓库中的第一条记录,并触发IDBRequest对象的onsuccess事件,显示这条记录。之后,调用cursor.continue()方法,将游标移动到下一条记录,并继续触发IDBRequest对象的onsuccess事件,显示这条记录,直到所有记录处理结束。

    <!--demo1108.html-->
    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange;
    var dbName = "myDb1";
    var idb;
    function init() {
    var dbConnect = indexedDB.open(dbName);
    dbConnect.onsuccess = function (e) {
    idb = e.target.result;
    }
    }
    function travelData() {
    var tx = idb.transaction(["students"], "readonly");
    var store = tx.objectStore("students");
    var result = document.getElementById("result");
    //var range=IDBKeyRange.bound(1,4);
    //var direction="next";
    //var req=store.openCursor(range,direction);
    var req = store.openCursor();
    req.onsuccess = function (e) {
    var cursor = this.result;
    if (cursor) {
    result.innerHTML += "id:" + cursor.key + " Name:" + cursor.value.sName + " Address:" + cursor.value.address + "<br/>";
    cursor.continue();
    }
    else {
    alert("finished");
    }
    }
    req.onerror = function (e) {
    alert("search error");
    }
    }
    </script>
    </head>
    <body onload="init()">
    <input type="button" value="Travel Data" onclick="travelData()"/>
    <hr/>
    <div id="result"></div>
    </body>
    </html>

 

B. 查找指定范围的记录

  如果希望遍历对象仓库中的部分数据,可以在openCurson()方法中设定查询范围。如果检索的数据不是对象仓库的主键,还需要使用对象仓库的index()方法获取当前要检索数据的索引。

 

    <!--demo1109.html-->
    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="UTF-8">
    <script>
    window.indexedDB = window.indexedDB || window.webkitIndexedDB;
    window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange;
    var dbName = "myDb1";
    var idb;
    function init() {
    var dbConnect = indexedDB.open(dbName);
    dbConnect.onsuccess = function (e) {
    idb = e.target.result;
    }
    }
    function searchData() {
    var result = document.getElementById("result"); //得到显示区域
    var tx = idb.transaction(["students"], "readonly");//开启事务
    var store = tx.objectStore("students"); //获取对象仓库
    var idx = store.index("ageIndex"); //获取当前索引
    var range = IDBKeyRange.only(18); //设置查询范围
    var direction = "next"; //设置查询方向
    var req = idx.openCursor(range, direction); //打开游标
    req.onsuccess = function (e) {
    var cursor = this.result;
    if (cursor) {
    result.innerHTML += "id:" + cursor.key + " Name:" + cursor.value.sName + " Address:" + cursor.value.address + "<br/>";
    cursor.continue();
    }
    else {
    alert("finished");
    }
    }
    req.onerror = function (e) {
    alert("search error");
    }
    }
    </script>
    </head>
    <body onload="init()">
    <input type="button" value="Search Data" onclick="searchData()"/>
    <hr/>
    <div id="result"></div>
    </body>
    </html>

QQ截图20220204231219.png

相关文章
|
2月前
|
存储 移动开发 大数据
HTML5 Web IndexedDB 数据库详解
IndexedDB 是一种高效的浏览器存储方案,允许在本地存储大量结构化数据,支持索引和事务,适用于需要离线和大数据处理的应用。它由数据库、对象仓库等组成,通过键值对存储数据,确保数据一致性和完整性。本介绍展示了如何创建、读取、更新和删除数据,以及事务和错误处理的最佳实践。
|
2月前
|
存储 移动开发 数据库
HTML5 Web IndexedDB 数据库常用数据存储类型
IndexedDB 支持多种数据存储类型,满足复杂数据结构的存储需求。它包括基本数据类型(如 Number、String、Boolean、Date)、对象(简单和嵌套对象)、数组、Blob(用于二进制数据如图像和视频)、ArrayBuffer 和 Typed Arrays(处理二进制数据)、结构化克隆(支持 Map 和 Set 等复杂对象),以及 JSON 数据。尽管不直接支持非序列化数据(如函数和 DOM 节点),但可以通过转换实现存储。开发者应根据具体需求选择合适的数据类型,以优化性能和使用体验。
|
2月前
|
SQL 存储 移动开发
HTML5 Web SQL 数据库详解
Web SQL 数据库是 HTML5 中的一种本地存储技术,允许在浏览器中使用 SQL 语言操作本地数据,支持离线访问和事务处理,适用于缓存数据和小型应用。然而,其存储容量有限且仅部分现代浏览器支持,标准已不再积极维护,未来可能被 IndexedDB 和 localStorage 等技术取代。使用时需谨慎考虑兼容性和发展前景。
|
4月前
|
前端开发 Java 关系型数据库
通过HTML网页对mysql数据库进行增删改查(CRUD实例)
通过HTML网页对mysql数据库进行增删改查(CRUD实例)
292 0
|
7月前
|
前端开发 Java 数据库
HTML、sqlserver数据库和java前七章节的复习(一)
HTML、sqlserver数据库和java前七章节的复习(一)
|
7月前
|
Java 数据库连接 数据库
基于SpringBoot+maven+Mybatis+html慢性病报销系统(源码+数据库)
基于SpringBoot+maven+Mybatis+html慢性病报销系统(源码+数据库)
113 0
|
存储 SQL NoSQL
浏览器端数据库IndexedDB
1.概念 IndexedDB是一种轻量级NoSQL(Not Only SQL,泛指非关系型)数据库,用来持久化大量客户端数据。
|
SQL 移动开发 API
HTML5也能操作数据库
HTML5也能操作数据库
111 0
|
2天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
10 3
|
2天前
|
安全 关系型数据库 MySQL
MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!
《MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!》介绍了MySQL中的三种关键日志:二进制日志(Binary Log)、重做日志(Redo Log)和撤销日志(Undo Log)。这些日志确保了数据库的ACID特性,即原子性、一致性、隔离性和持久性。Redo Log记录数据页的物理修改,保证事务持久性;Undo Log记录事务的逆操作,支持回滚和多版本并发控制(MVCC)。文章还详细对比了InnoDB和MyISAM存储引擎在事务支持、锁定机制、并发性等方面的差异,强调了InnoDB在高并发和事务处理中的优势。通过这些机制,MySQL能够在事务执行、崩溃和恢复过程中保持
13 3