IndexedDB:
步骤:
1. 打开数据库:request=indexedDB.open(dbName,1)
2. 写打开失败的方法:request.onerror=function(e){……};
3. 写打开成功的方法:request.onsuccess=function(e){……};
4. 写数据库版本更新时执行的方法:request.onupgradeneeded =function(event){……}
5. 在第4步的方法里创建数据对象仓库:objectStore = db.createObjectStore("仓库名", {keyPath:"主键",autoIncrement:true自增|false不自增});
6. 在第5步后面创建索引:var idx = store.createIndex('usernameIndex','userName',{unique:false})
7. 在第6步以后,创建一个事务数据库,var tx = db.transaction('对象仓库名','事务方法readonly只读|readwrite读写');事务数据库可以使用这三个事件:onabort():事务中断。onomplete():事务完成。onerror():事务出错。
8. 在第3步,展开写,db=event.target.result;
9. 对数据进行操作,在第7步后面写var store = tx.objectStore("test");var ob = store.get(x);方法有add(),put(),delete()get()
10. 如果要对数据遍历的话接着第9步:var store = tx.objectStore("test");var cursor =store.openCursor();
IndexedDB 是一种可以让你在用户的浏览器内持久化存储数据的方法。IndexedDB 为生成 Web Application 提供了丰富的查询能力,使我们的应用在在线和离线方法时都可以正常工作。
基本模式
IndexedDB 鼓励使用的基本模式如下所示:
1. 打开数据库并且开始一个事务。
2. 创建一个object store。
3. 构建一个请求来执行一些数据库操作,像增加或提取数据等。
4. 通过监听正确类型的 DOM 事件以等待操作完成。
5. 在操作结果上进行一些操作(可以在 request 对象中找到)
IndexedDB具有以下特点:
(1)键值对储存。 IndexedDB内部采用对象仓库(objectstore)存放数据。所有类型的数据都可以直接存入,包括JavaScript对象。在对象仓库中,数据以“键值对”的形式保存,每一个数据都有对应的键名,键名是独一无二的,不能有重复,否则会抛出一个错误。
(2)异步。 IndexedDB操作时不会锁死浏览器,用户依然可以进行其他操作,这与localStorage形成对比,后者的操作是同步的。异步设计是为了防止大量数据的读写,拖慢网页的表现。
(3)支持事务。 IndexedDB支持事务(transaction),这意味着一系列操作步骤之中,只要有一步失败,整个事务就都取消,数据库回到事务发生之前的状态,不存在只改写一部分数据的情况。
(4)同域限制IndexedDB也受到同域限制,每一个数据库对应创建该数据库的域名。来自不同域名的网页,只能访问自身域名下的数据库,而不能访问其他域名下的数据库。
(5)储存空间大 IndexedDB的储存空间比localStorage大得多,一般来说不少于250MB。IE的储存上限是250MB,Chrome和Opera是剩余空间的某个百分比,Firefox则没有上限。
(6)支持二进制储存。
判断是否支持indexedDB:
if (!window.indexedDB){
window.alert("Your browser doesn't support a stable version ofIndexedDB. Such and such feature will not be available.")
}
// 打开我们的数据库
var request = indexedDB.open('dbName',1); // 打开 dbName 数据库
request.onerror= function(e){ // 监听连接数据库失败时执行
console.log('连接数据库失败');
}
request.onsuccess= function(e){ // 监听连接数据库成功时执行
console.log('连接数据库成功');
}
indexedDB.open 方法来连接数据库,该方法接收两个参数,第一个是数据库名,第二个是数据库版本号。该方法会返回一个IDBOpenDBRequest对象,代表一个请求连接数据库的请求对象。我们可以通过监听请求对象的 onsuccess 和 onerror 事件来定义连接成功或失败需执行的方法。
var request = indexedDB.open('dbName',2); // 更新版本,打开版本为2的数据库
// ...
request.onupgradeneeded = function(e){
console.log('新数据库版本号为=' + e.newVersion);
}
我们通过监听请求对象的onupgradeneeded事件来定义数据库版本更新时执行的方法。
第一次打开数据库时,会先触发upgradeneeded事件,然后触发success事件。
创建对象仓库
object store(对象仓库)是 indexedDB 数据库的基础,在indexedDB中并没有数据库表,而对象仓库,就是相当于一个数据库表。
var request = indexedDB.open('dbName', 3);
// ...
request.onupgradeneeded = function(e){
var db = e.target.result;//要有的
var store = db.createObjectStore('Users', {keyPath: 'userId',autoIncrement: false});
console.log('创建对象仓库成功');
}
db.createObjectStore 方法接收两个参数,第一个为对象仓库名,第二个参数为可选参数,值为一个js对象。该对象中的 keyPath 属性为主键,相当于数据库表中 id 为主键。autoIncrement 属性为 false,则表示主键值不自增,添加数据时需指定主键值。
注意:在数据库中,对象仓库名不可重复,否则浏览器会报错。
Key Path Key Generator Description
1. No No :这种对象存储空间可以持有任意类型的值,甚至是像数字和字符串这种基本数据类型的值。每当我们想要增加一个新值的时候,必须提供一个单独的键参数。
2. Yes No :这种对象存储空间只能持有 JavaScript对象。这些对象必须具有一个和 key path 同名的属性。
3. No Yes :这种对象存储空间可以持有任意类型的值。键会为我们自动生成,或者如果你想要使用一个特定键的话你可以提供一个单独的键参数。
4. Yes Yes :这种对象存储空间只能持有 JavaScript对象。通常一个键被生成的同时,生成的键的值被存储在对象中的一个和 key path 同名的属性中。然而,如果这样的一个属性已经存在的话,这个属性的值被用作键而不会生成一个新的键。
objectStoreNames属性(是db的属性)
objectStoreNames属性返回一个DOMStringList对象,里面包含了当前数据库所有“对象仓库”的名称。可以使用DOMStringList对象的contains方法,检查数据库是否包含某个“对象仓库”。
if(!db.objectStoreNames.contains("firstOS")){//上面代码先判断某个“对象仓库”是否存在,如果不存在就创建该对象仓库。
db.createObjectStore("firstOS");
}
创建索引
indexedDB 数据库中通过数据对象的某个属性来创建索引,在数据库中进行检索时,只能通过被设为索引的属性进行检索。
var request = indexedDB.open('dbName', 4);
// ...
request.onupgradeneeded = function(e){
var db = e.target.result;
var store = db.createObjectStore('newUsers', {keyPath: 'userId',autoIncrement: false});
var idx = store.createIndex('usernameIndex','userName',{unique: false})
console.log('创建索引成功');
}
store.createIndex 方法接收三个参数,第一个为索引名,第二个为数据对象的属性,上例中使用 userName 属性来创建索引,第三个参数为可选参数,值为一个js对象。该对象中的 unique 属性为 true,代表索引值不可以相同,即两条数据的userName不可以相同,为 false 则可以相同。
transaction方法是db的方法
transaction方法用于创建一个数据库事务。向数据库添加数据之前,必须先创建数据库事务。
在indexedDB 中,所有数据操作都只能在事务中执行。连接数据库成功后,可以使用IDBOpenDBRequest 对象的transaction方法开启只读事务或读写事务。
transaction方法有三个事件,可以用来定义回调函数。
1. onabort():事务中断。
2. onomplete():事务完成。
3. onerror():事务出错。
var request = indexedDB.open('dbName', 5);
// ...
request.onupgradeneeded = function(e){
var db = e.target.result;
var tx = db.transaction('Users','readonly');
tx.oncomplete = function(e){
console.log('事务结束了');
}
tx.onabort = function(e){
console.log('事务被中止了');
}
}
db.transaction 方法接收两个参数,第一个参数可以是字符串或数组,字符串时则是一个对象仓库名,数组时则是由对象仓库名组成的数组,transaction 可以对参数中任何一个对象仓库进行操作。第二个参数为事务模式,传入readonly时只能对对象仓库进行读操作,无法写操作。可以传入 readwrite进行读写操作。
transaction方法返回一个事务对象,该对象的objectStore方法用于获取指定的对象仓库。
var t = db.transaction(["firstOS"],"readwrite");
var store =t.objectStore("firstOS");
操作数据(的方法)
1. add() : 增加数据。接收一个参数,为需要保存到对象仓库中的对象。
2. put() : 增加或修改数据。接收一个参数,为需要保存到对象仓库中的对象。
3. get() : 获取数据。接收一个参数,为需要获取数据的主键值。
4. delete() : 删除数据。接收一个参数,为需要获取数据的主键值。
var request =indexedDB.open('dbName', 5);
// ...
request.onsuccess= function(e){
var db = e.target.result;
var tx = db.transaction('Users','readwrite');
var store = tx.objectStore('Users');
var value = {
'userId': 1,
'userName': 'linxin',
'age': 24
}
var req1 = store.put(value); // 保存数据
var req2 = store.get(1); // 获取索引userId为1的数据
req2.onsuccess = function(){
console.log(this.result.userName); // linxin
}
var req3 = store.delete(1); // 删除索引为1的数据
req3.onsuccess = function(){
console.log('删除数据成功'); // 删除数据成功
}
}
add 和 put 的作用类似,区别在于 put 保存数据时,如果该数据的主键在数据库中已经有相同主键的时候,则会修改数据库中对应主键的对象,而使用 add 保存数据,如果该主键已经存在,则保存失败。
检索数据
上面我们知道使用 get() 方法可以获取数据,但是需要制定主键值。如果我们想要获取一个区间的数据,可以使用游标。游标通过对象仓库的openCursor方法创建并打开。
想要遍历数据,就要openCursor方法,它在当前对象仓库里面建立一个读取光标(cursor)。
var t = db.transaction(["test"],"readonly");
var store =t.objectStore("test");
var cursor = store.openCursor();
openCursor方法也是异步的,有自己的success和error事件,可以对它们指定回调函数。
cursor.onsuccess = function(e) {
var res = e.target.result;
if(res) {
console.log("Key", res.key);
console.dir("Data", res.value);
res.continue();
}
}
openCursor 方法接收两个参数,
第一个是 IDBKeyRange 对象,该对象创建方法主要有以下几种:
IDBKeyRange对象的作用是生成一个表示范围的Range对象。生成方法有四种:
1. lowerBound方法:指定范围的下限。
2. upperBound方法:指定范围的上限。
3. bound方法:指定范围的上下限。
4. only方法:指定范围中只有一个值。
// boundRange 表示主键值从1到10(包含1和10)的集合。
// 如果第三个参数为true,则表示不包含最小键值1,如果第四参数为true,则表示不包含最大键值10,默认都为false
var boundRange = IDBKeyRange.bound(1, 10,false, false);
// onlyRange 表示由一个主键值的集合。only() 参数则为主键值,整数类型。
var onlyRange = IDBKeyRange.only(1);
// lowerRaneg 表示大于等于1的主键值的集合。
// 第二个参数可选,为true则表示不包含最小主键1,false则包含,默认为false
var lowerRange = IDBKeyRange.lowerBound(1,false);
// upperRange 表示小于等于10的主键值的集合。
// 第二个参数可选,为true则表示不包含最大主键10,false则包含,默认为false
var upperRange = IDBKeyRange.upperBound(10,false);
openCursor 方法的第二个参数表示游标的读取方向,主要有以下几种:
1. next : 游标中的数据按主键值升序排列,主键值相等的数据都被读取
2. nextunique : 游标中的数据按主键值升序排列,主键值相等只读取第一条数据
3. prev : 游标中的数据按主键值降序排列,主键值相等的数据都被读取
4. prevunique : 游标中的数据按主键值降序排列,主键值相等只读取第一条数据
var request = indexedDB.open('dbName', 6); //打开一个数据库,第6的版本
// ...
request.onsuccess = function(e){//打开并创建一个游标
var db = e.target.result;
var tx = db.transaction('Users','readwrite');
var store = tx.objectStore('Users');
var range = IDBKeyRange.bound(1,10);
var req = store.openCursor(range, 'next');
req.onsuccess = function(){
var cursor = this.result;
if(cursor){
console.log(cursor.value.userName);
cursor.continue();
}else{
console.log('检索结束');
}
}
}
当存在符合检索条件的数据时,可以通过 update 方法更新该数据:
cursor.update({
userId : cursor.key,
userName : 'Hello',
age : 18
});
可以通过 delete 方法删除该数据:
cursor.delete();
可以通过 continue 方法继续读取下一条数据,否则读到第一条数据之后不再继续读取:
cursor.continue();
例子:
<!DOCTYPE html>
<html>
<head>
<metaname="viewport" content="width=device-width,initial-scale=1" />
<title>HTML5- IndexedDB</title>
<scripttype="text/javascript">
var db;
var request;//打开的数据库名字
var objectStore;//对象仓库
//创建数据库
function _create(dbName){
request=indexedDB.open(dbName,1);//打开dbName数据库,版本为1;
//监听链接数据库失败
request.onerror = function () {
alert("打开数据库失败:"+event.target.message);
//监听链接数据库失败,执行的函数
}
//监听连接数据库成功执行
request.onsuccess = function (event) {//监听连接数据库成功
alert("打开数据库成功!");
db=event.target.result;//这一句一定要加,什么意思,不理解
var transaction = db.transaction(["info"],"readwrite");
objectStore = transaction.objectStore("info");
}
//定义数据库版本更新时执行的方法
request.onupgradeneeded = function(event) {
alert("版本变化!");
db = event.target.result;
//创建对像仓库,info:对象仓库名,主键是userId,autoincrement:true主键自增,false为不自增;
objectStore = db.createObjectStore("info", {keyPath:"userId",autoIncrement: true});
}
}
//删除数据库
function _delete(dbName){
try{
request=indexedDB.deleteDatabase(dbName);
request.onerror = function (event) {
alert("删除数据库失败:"+event.target.message);
}
request.onsuccess = function (event) {
alert("删除数据库成功!");
}
}catch(e){
alert(e.getMessage);
}
}
//获取数据库
function _get (argument) {
var p=document.getElementById("display");//获得html中的display
p.innerHTML="";//获取数据前先清理一下页面已显示的数据
if(!db){
alert("请打开数据先!");
return false;
}
var store =db.transaction("info").objectStore("info");
var keyRange = IDBKeyRange.lowerBound(0);//规定keyRange从0开始
var cursorRequest = store.openCursor(keyRange);//按照keyRange的设置开启游标
cursorRequest.onsuccess = function (e) {
var result = e.target.result;
if (!!result == false)
return;
_render(result.value);
result.continue();//这边执行轮询读取
};
cursorRequest.onerror = function (e) {
alert("数据检索失败!");
};
}
function _render (e) {
var p=document.getElementById("display");
p.innerHTML+="姓名:"+e.name+" 年龄:"+e.age+" 性别:"+e.xb+"<br />";
}
function _add (argument) {
var name=document.getElementById("name").value;
var age=document.getElementById("age").value;
var xb,flag=document.getElementById("nan").checked;
if(flag)
xb="男";
else
xb="女";
var detail={
name:name,
age:age,
xb:xb
}
if(!db){
alert("请打开数据先!");
return false;
}
var transaction = db.transaction(["info"],"readwrite");//所有数据操作都只能在事务中执行
//info:是对象名仓库,readwrite是读写事务
objectStore = transaction.objectStore("info");
objectStore.add(detail);//给数据仓库添加这条数据
alert("添加成功!");
}
</script>
</head>
<body>
<small>添加前请先打开数据库(如数据库不存在则执行创建过程)</small><br /><br />
<button οnclick="_create('user')">打开数据库</button>
<button οnclick="_delete('user')">删除数据库</button><br/><br />
姓名:<input type="text" id="name"><br />
年龄:<input type="number" id="age" min=1><br/>
性别:<br />
男:<inputtype="radio" id="nan" name="xb"value="male" checked>
女:<inputtype="radio" id="nv" name="xb"value="female"><br />
<button οnclick="_add()">添加数据</button>
<button οnclick="_get()">获取数据</button><br/>
<p id="display"></p>
</body>
</html>
我也是刚学,这是我用自己的理解做的笔记的。
说的不一定准确
其实我也很菜的啦