IndexedDB是浏览器提供的本地数据库,它可以被网页脚本创建和操作。IndexedDB 允许储存大量数据,提供查找接口,还能建立索引。就数据库类型而言,IndexedDB是非关系型数据库(不支持 SQL 查询语句),更接近NoSQL数据库,并且Web Worker中可用。

IndexedDB的特点

  1. 键值对储存:IndexedDB内部采用对象仓库(object store)存放数据。所有类型的数据都可以直接存入,包括 JavaScript 对象。对象仓库中,数据以"键值对"的形式保存,每一个数据记录都有对应的主键,主键是独一无二的,不能有重复,否则会抛出一个错误。
  2. 异步:不同于LocalStorage,IndexedDB操作是异步的。异步设计是为了防止大量数据的读写时浏览器卡死。
  3. 支持事务:IndexedDB支持事(transaction),这意味着一系列操作步骤之中,只要有一步失败,整个事务就都取消,数据库回滚到事务发生之前的状态,不存在只改写一部分数据的情况。
  4. 同源限制:IndexedDB 受到同源限制,每一个数据库对应创建它的域名。网页只能访问自身域名下的数据库,而不能访问跨域的数据库。
  5. 储存空间大:IndexedDB的储存空间非常大。
  6. 支持二进制储存:IndexedDB不仅可以储存字符串,还可以储存二进制数据(ArrayBuffer 对象和 Blob 对象)。

存储限制

浏览器的最大存储空间是动态的,取决于硬盘大小。全局限制为可用磁盘空间的50%。假设可用硬盘驱动器是500GB,那么浏览器的总存储容量为250GB。如果超过此范围,则会发起称为源回收的过程,删除整个源的数据,直到存储量再次低于限制。删除源数据没有只删一部分的说法——因为这样可能会导致不一致的问题。

还有另一个限制称为组限制(全局限制的20%)。最小组限制为10MB,但组限制不能超过全局限制,因此如果内存非常低,比如全局限制为8MB,则组限制也将为8MB。最大组限制为2GB。 每个源都是一组(源组)的一部分,每个顶级域名都有一个组,每个子域名都有一个源。

在隐私浏览模式下,大多数数据存储不被支持。本地存储数据和cookie仍然可用,但它们是短暂的,当关闭最后一个隐私浏览窗口时,数据将被删除。

Chrome

Internet Explorer

Firefox

Safari

允许浏览器使用多达80%的总磁盘空间。一个来源最多可以使用总磁盘空间的 60%。其他基于 Chromium 的浏览器可能允许浏览器使用更多的存储空间。

Internet Explorer 10 及更高版本最多可以存储250MB,并且在使用量超过10MB时会提示用户

允许浏览器使用多达50%的可用磁盘空间,每个源组最多可以使用2GB。

允许 1GB 左右。当达到限制时,Safari会提示用户,同时以200MB的增量增加限制。

使用StorageManager API可以查询可用存储量及其目前使用的存储量,它会报告 IndexedDB和缓存API使用的总字节数,还可以计算近似的可用剩余存储空间。Chrome 则会始终报告实际磁盘大小的60%,其他基于 Chromium的浏览器在报告可用配额时可能会考虑可用空间量。

if (navigator.storage && navigator.storage.estimate) {
    const quota = await navigator.storage.estimate();
    // quota.usage -> 已用字节数。
    // quota.quota -> 最大可用字节数。
    const percentageUsed = (quota.usage / quota.quota) * 100;
    console.log(`您已使用可用存储的 ${percentageUsed}%。`);
    const remaining = quota.quota - quota.usage;
    console.log(`您最多可以再写入 ${remaining} 个字节。`);
}

StorageManager并不是所有浏览器都支持。

打开数据库

使用indexedDB.open打开数据库,open方法返回一个IDBOpenDBRequest对象,同时这是一个异步操作,open操作并不会立马打开数据库或者开启事务,我们可以通过监听request的事件来进行相应的处理。如果要打开的数据库不存在,浏览器会自动创建。

open方法传入两个参数,第一个参数是数据库的名字,第二个参数是数据库的版本号。

当创建一个新的数据库或者升级一个现有的数据库版本的时候,将会触发一个onupgradeneeded事件,并在事件中传入IDBVersionChangeEvent,我们可以通过event.target.result来获取到IDBDatabase对象,然后通过这个对象来进行数据库的版本升级操作。

const version = 1; // 版本主要用来控制数据库的结构,当数据库结构(表结构)发生变化时,版本也会变化
const dbName = "myData";
const request = window.indexedDB.open(dbName, version);

let db; // 全局 IndexedDB 数据库实例

// onupgradeneeded 在版本改变时触发(首次连接数据库时,版本从 0 变成 1,因此也会触发,且先于 onsuccess)
request.onupgradeneeded = (event) => {
    db = event.target.result;
};

// onsuccess 在连接成功后触发
request.onsuccess =  (event) => {
    db = request.result;
    console.log("db connected");
};

// onerror 在连接失败时触发
request.onblocked =  (event) => {
    console.log("db request blocked!");
}

// onblocked 在连接被阻止的时候触发,比如打开版本低于当前存在的版本
request.onerror =  (event) => {
    console.log("error!");
};

创建数据表

主键(key)是默认建立索引的属性,既可以让IndexedDB自动生成主键,也可以使用数据记录里面合适的属性作为主键。比如,数据记录是{ id: 1, name: ‘张三’ },那么id属性可以作为主键。主键也可以指定为下一层对象的属性,比如{ foo: { bar: ‘baz’ } }的foo.bar也可以指定为主键。

request.onupgradeneeded = (event) => {
    const db = event.target.result;
    if (!db.objectStoreNames.contains("person")) {
        // 创建数据表并设置id为自增主键
        db.createObjectStore("person", {
            "keyPath": "id", // 主键
            "autoIncrement": true // 自增(每加一条数据,主键会自动增长,无需开发者指定)
        });
    }
};

添加数据

在indexedDB里任何的存取对象的操作都需要放在事务里执行。为了往数据库里新增数据,我们首先需要创建一个事务,新建时必须指定表格名称和操作模式(读写权限)。

新建事务以后,通过IDBTransaction.objectStore(name)方法,拿到 IDBObjectStore 对象,再通过表格对象的add()方法,向表格写入一条记录。

写入操作是一个异步操作,通过监听连接对象的success事件和error事件,了解是否写入成功。

const transaction = db.transaction(["person"], "readwrite"); // 创建事务
const objectStore = transaction.objectStore("person"); // 指定person表
objectStore.add({ 
    "name": "张三", 
    "age": 24, 
    "email": "zhangsan@example.com"
}); // 添加数据

transaction.oncomplete = (event) => {
    console.log("数据写入成功");
};

transaction.onerror = (event) => {
    console.log("数据写入失败");
};

获取数据

读取数据也是通过事务完成,需要先创建一个readonly的Transaction,通过IDBTransaction.objectStore(name)方法,拿到IDBObjectStore 对象,再通过表格对象的get()方法读取数据,参数是主键的值。

const transaction = db.transaction(["person"], "readonly");
const objectStore = transaction.objectStore("person");
const request = objectStore.get(1);

request.onerror = (event) => {
    console.log("事务失败");
};

request.onsuccess = (event) => {
    const result = event.target.result;
    if (result) {
        console.log("Name: " + result.name);
        console.log("Age: " + result.age);
        console.log("Email: " + result.email);
    } else {
        console.log("未获得数据记录");
    }
};

遍历数据

遍历数据表格的所有记录,要使用指针对象IDBCursor。

const transaction = db.transaction(["person"], "readonly");
const objectStore = transaction.objectStore("person");
const request = objectStore.openCursor();

// penCursor()方法是一个异步操作,要监听success事件
request.onsuccess = (event) => {
    const cursor = event.target.result;
    if (cursor) {
        console.log("Id: " + cursor.key);
        console.log("Name: " + cursor.value.name);
        console.log("Age: " + cursor.value.age);
        console.log("Email: " + cursor.value.email);
        cursor.continue()
    } else {
        console.log("没有更多数据了!");
    }
};

更新数据

更新数据要使用IDBObject.put()方法。

const transaction = db.transaction(["person"], "readwrite");
const objectStore = transaction.objectStore("person");
const request = objectStore.put({
    "id": 1,
    "name": "李四", 
    "age:" 35, 
    "email": "lisi@example.com"
}); // put()方法自动更新了主键为1的记录

request.onerror = (event) => {
    console.log("数据更新失败");
};

request.onsuccess = (event) => {
    console.log("数据更新成功");
};

删除数据

IDBObjectStore.delete()方法用于删除记录。

const transaction = db.transaction(["person"], "readwrite");
const objectStore = transaction.objectStore("person");
const request = objectStore.delete(1);

request.onerror = (event) => {
    console.log("数据删除失败");
};

request.onsuccess = (event) => {
    console.log("数据删除成功");
};

新建索引

IDBObject.createIndex()的三个参数分别为索引名称、索引所在的属性、配置对象(说明该属性是否包含重复的值)

request.onupgradeneeded = (event) => {
    const db = event.target.result;
    if (!db.objectStoreNames.contains("person")) {
        // 创建数据表并设置id为自增主键
        const objectStore = db.createObjectStore("person", {
            "keyPath": "id", // 主键
            "autoIncrement": true // 自增(每加一条数据,主键会自动增长,无需开发者指定)
        });
        // 新建对象仓库以后,下一步可以新建索引
        objectStore.createIndex("name", "name", { 
            "unique": false 
        });
        objectStore.createIndex("email", "email", { 
            "unique": true 
        });
    }
});

查询索引

索引的意义在于,可以让你搜索任意字段,也就是说从任意字段拿到数据记录。如果不建立索引,默认只能搜索主键(即从主键取值)。

const transaction = db.transaction(["person"], "readonly");
const objectStore = transaction.objectStore("person");
const index = objectStore.index("name");
const request = index.get("李四");

request.onerror = (event) => {
    console.log("数据查询失败");
};

request.onsuccess = (event) => {
    const result = event.target.result;
    if (result) {
        console.log("Name: " + result.name);
        console.log("Age: " + result.age);
        console.log("Email: " + result.email);
    } else {
        console.log("未获得数据记录");
    }
};

删除数据库

indexedDB.deleteDatabase()方法用于删除一个数据库,参数为数据库的名字。它会立刻返回一个IDBOpenDBRequest对象,然后对数据库执行异步删除。删除操作的结果会通过事件通知,IDBOpenDBRequest对象可以监听以下事件。

const dbName = "myData";
const request = window.indexedDB.deleteDatabase(dbName);

request.onerror = (event) => {
    console.log("删除失败");
};

request.onsuccess = (event) => {
    console.log("删除成功");
};

相关库

  • Godb.js让你即使你不了解浏览器数据库sIndexedDB也能使用 ,从而把关注点放到业务上面去。
  • localForage一个简单的 Polyfill,提供了简单的客户端数据存储的值语法。它在后台使用IndexedDB,并在不支持 IndexedDB的浏览器中回退到WebSQL或localStorage。
  • Dexie.jsIndexedDB 的包装,通过简单的语法,可以更快地进行代码开发。
  • ZangoDB类似MongoDB的IndexedDB接口,支持MongoDB的大多数熟悉的过滤、投影、排序、更新和聚合功能。
  • JsStore一个带有SQL语法的IndexedDB包装器。
  • MiniMongo由localstorage支持的客户端内存中的mongodb,通过http进行服务器同步。MeteorJS使用MiniMongo。
  • PouchDB使用IndexedDB在浏览器中实现CouchDB的客户端。
  • idb一个微小的(~1.15k)库,大多API与IndexedDB类似,但做了一些小的改进,让数据库的可用性得到了大大的提升。
  • idb-keyval使用IndexedDB实现的超级简单且小巧的(~600B)基于Promise的键值对存储。
  • sifrr-storage一个非常小的(~2kB)基于Promise的客户端键值数据库。基于 IndexedDB、localStorage、WebSQL和Cookies 实现。它可以自动选择上述支持的数据库,并按照优先顺序使用。
  • lovefieldLovefield是一个用于Web App 的关系型数据库,使用JavaScript编写,可以在不同的浏览器环境中运行,提供了类似SQL的 API,速度快、安全且易用。

参考:IndexedDB API