Обзор
В этом документе принимаетсяIndexedDB
Представлены методы использования и сценарии использования, а также даны ответы на часто задаваемые вопросы.
В то же время, поскольку в соответствующих документах в MDN отсутствует соответствующая логика, их нелегко понять. В этой статье контент будет организован с помощью общих требований к хранению данных и операционных требований в проектах.
Читатели могут научиться правильно использовать его в проекте с помощью этой статьи.IndexedDB
, предоставляя приложениям возможности локального хранения и избегая некоторых распространенных проблем.
Причина: разработчикам необходимо постоянно хранить локально
Когда мы разработаем несколько больших страниц SPA, нам нужно будет хранить некоторые данные локально.
Когда объем данных невелик, мы можем использовать SessionStorage или LocalStorage для его хранения, но когда объем данных велик или соответствует определенным спецификациям, мы можем использовать базу данных для хранения данных.
В базе данных браузера естьweb sql
иIndexedDB
два вида. По сравнению с HTML5 устарелweb sql
Рекомендуется использоватьIndexedDB
.
структура
Ниже мы используем изображение, чтобы понять общую структуру IndexedDB.
аналогияsql
тип базы данных,IndexedDB
БД (база данных) вsql
в БД, покаObject Store(存储空间)
является数据表
,Item
равно записи в таблице记录
.
Использование ИндекседБД
Теперь мы основываемся наIndexedDB
Структура хранилища представлена для ознакомления с его работой, чтобы каждый мог иметь предварительное представление об этом пространстве хранения. В основном мы представляем:
- операции с базой данных
- Работа с таблицей данных
- манипуляция данными
операции с базой данных
Создать или открыть базу данных
использоватьIndexedDB
Первым шагом является создание или открытие базы данных. Мы используемwindow.indexedDB.open(DBName)
Этот API для вызова для работы. Конкретные примеры следующие:
const request = window.indexedDB.open('test');
request.onupgradeneeded = function (event) {
}
request.onsuccess = function(event) {
//request === event.target;
}
request.onerror = function(event) {}
При вызове этого интерфейса, если текущая база данных не существует, будет создана новая база данных.
Когда соединение с базой данных установлено,IDBOpenDBRequest
объект.
Когда соединение установлено успешно, оно активируетonsuccess
событие, где параметр функцииevent
изtarget
собственностьrequest
объект.
Когда база данных будет создана или версия будет обновлена, она сработаетonupgradeneeded
событие.
Обновить номер версии базы данных
window.indexedDB.open
Второй параметр — это номер версии. Если не указано, номер версии по умолчанию — 1. Конкретные примеры следующие:
const request = window.indexedDB.open('test', 2);
Когда необходимо обновить базу данныхschema(模式)
, вам необходимо обновить номер версии. На этом этапе мы указываем номер версии выше, чем предыдущая версия, которая вызоветonupgradeneeded
событие. Точно так же, когда база данных не существует, это событие также будет запущено, и версия будет обновлена до верхней версии.
Мы должны отметить, что номер версииUnsigned long long
число, что означает, что это может быть очень большое целое число. Однако это не может быть десятичное число, иначе оно будет округлено до ближайшего целого числа, что может привести кonUpgradeneeded
Событие не срабатывает (баг).
Операции с хранилищем
Создать место для хранения
Мы используемcreateObjectStore
для создания места для хранения. использовать одновременноcreateIndex
для создания его индекса. Конкретные примеры следующие:
var request = window.indexedDB.open('test', 1);
request.onupgradeneeded = function (event) {
var db = event.target.result;
var objectStore = db.createObjectStore('table1', {keyPath: 'id', autoIncrement: true});
objectStore.createIndex('name', 'name', {unique: false});
}
request.onerror = function (event) {
alert("Why didn't you allow my web app to use IndexedDB?!");
};
Примечание: доступно только вonupgradeneeded
Место для хранения создается в функции обратного вызова, а не после открытия базы данных.success
Создается в функции обратного вызова.
пройти черезcreateObjectStore
Возможность создания места для хранения. Принимает два параметра:
- Первый параметр, имя хранилища, которое у нас выше
customers
. - Второй параметр указывает сохраненное
keyPath
Значение является атрибутом объекта хранилища, которое можно использовать в качестве значения ключа при получении данных о пространстве хранения.autoIncrement
указанныйkey
Является ли значение автоматически увеличивающимся (когда значение ключа является целым числом от 1 до 2^53 по умолчанию).
иcreateIndex
Можно установить индекс для текущего пространства хранения. Он принимает три параметра:
- Первый параметр, имя индекса.
- Второй параметр указывает, по какому атрибуту сохраненных данных строить индекс.
- Третье свойство, объект параметров, где свойство
unique
значениеtrue
Указывает, что значения индекса не могут быть равными.
манипуляция данными
дела
существуетIndexedDB
В , мы также можем использовать транзакции для выполнения операций с базой данных. Транзакции имеют три режима (константы устарели):
-
readOnly
, только чтение. -
readwrite
, прочти и напиши. -
versionchange
, версия базы данных меняется.
Когда мы создаем транзакцию, нам нужно выбрать режим из вышеперечисленных, если он не указан, по умолчанию используется режим только для чтения. Конкретные примеры следующие:
const transaction = db.transaction(['customers'], 'readwrite');
функция транзакцииtransaction
Первый параметр — это пространство для хранения, которое необходимо связать, а второй необязательный параметр — это режим транзакции. Подобно предыдущему, он также срабатывает при успешной транзакции.onsuccess
функция, срабатывающая при сбоеonerror
функция.
Транзакционные операции атомарны.
добавить данные
Когда пространство для хранения инициализировано, мы можем поместить данные в пространство для хранения. позвонить напрямуюadd
метод может поместить данные в пространство для хранения, конкретный пример выглядит следующим образом:
var request = window.indexedDB.open('test', 1);
request.onsuccess = function (event) {
var db = event.target.result;
var transaction = db.transaction(['table1'], 'readwrite');
var objectStore = transaction.objectStore('table1');
var index = objectStore.index('name');
objectStore.add({name: 'a', age: 10});
objectStore.add({name: 'b', age: 20});
}
Примечание:add
Значение ключа второго параметра в методе находится в указанном пространстве памяти.keyPath
значение, еслиdata
содержитkeyPath
value или это значение самоувеличивается, то этот параметр можно опустить.
Найти данные
Получить данные по определенному значению
Когда нам нужно получить данные из хранилища, мы можем использовать следующие методы:
var request = window.indexedDB.open('test', 1);
request.onsuccess = function (event) {
var db = event.target.result;
var transaction = db.transaction(['table1'], 'readwrite');
var objectStore = transaction.objectStore('table1');
var request = objectStore.get(1);
request.onsuccess = function (event) {
// 对 request.result 做些操作!
console.log(request.result);
};
request.onerror = function (event) {
// 错误处理!
};
}
Получить данные через курсор
Когда вам нужно просмотреть данные во всем пространстве хранения, вам нужно использовать курсор. Использование курсора выглядит следующим образом:
var request = window.indexedDB.open('test', 1);
request.onsuccess = function (event) {
var db = event.target.result;
var transaction = db.transaction(['table1'], 'readwrite');
var objectStore = transaction.objectStore('table1');
var request = objectStore.openCursor();
request.onsuccess = function (event) {
var cursor = event.target.result;
if (cursor) {
// 使用Object.assign方法是为了避免控制台打印时出错
console.log(Object.assign(cursor.value));
cursor.continue();
}
};
request.onerror = function (event) {
// 错误处理!
};
}
При использовании курсоров есть одно предостережение: когда курсор пересекает все пространство хранения, но не находит значения для заданного условия, он все равно сработает.onsuccess
функция.
openCursor
иopenKeyCursor
Есть два параметра:
-
Первый параметр, диапазон обхода, указывает диапазон доступа курсора. Диапазон проходит через
IDBKeyRange
способ получения параметров.Ниже приведены конкретные примеры параметров диапазона перемещения:
// 匹配值 key === 1
const singleKeyRange = IDBKeyRange.only(1);
// 匹配值 key >= 1
const lowerBoundKeyRange = IDBKeyRange.lowerBound(1);
// 匹配值 key > 1
const lowerBoundOpenKeyRange = IDBKeyRange.lowerBound(1, true);
// 匹配值 key < 2
const upperBoundOpenKeyRange = IDBKeyRange.upperBound(2, true);
// 匹配值 key >= 1 && key < 2
const boundKeyRange = IDBKeyRange.bound(1, 2, false, true);
index.openCursor(boundKeyRange).onsuccess = function(event) {
const cursor = event.target.result;
if (cursor) {
// Do something with the matches.
cursor.continue();
}
};
-
Второй параметр, порядок обхода, указывает порядок, в котором перемещается курсор, и метод обработки, когда повторяется один и тот же идентификатор (поле, заданное атрибутом keyPath). изменить область действия определенной строкой (
IDBCursor
константы устарели) для получения. в:-
next
, получить все данные от начала до конца (включая дубликаты) -
prev
, извлекает все данные сзади наперед (включая дубликаты) -
nextunique
, получить данные спереди назад (за дублирующиеся данные берется только первая запись, а если индекс повторяется, то он считается повторяющимся, то же самое ниже) -
prevunique
, получить данные сзади наперед (за дублирующиеся данные берется только первый)
Конкретный пример параметра порядка обхода выглядит следующим образом:
-
var request = window.indexedDB.open('test', 1);
request.onsuccess = function (event) {
var db = event.target.result;
var transaction = db.transaction(['table1'], 'readwrite');
var objectStore = transaction.objectStore('table1');
var lowerBoundOpenKeyRange = IDBKeyRange.lowerBound(1, false);
var request = objectStore.openCursor(lowerBoundOpenKeyRange, IDBCursor.PREV);
request.onsuccess = function (event) {
var cursor = event.target.result;
if (cursor) {
// 使用Object.assign方法是为了避免控制台打印时出错
console.log(Object.assign(cursor.value));
cursor.continue();
}
};
request.onerror = function (event) {
// 错误处理!
};
}
использовать индекс
Ранее при построении базы данных мы создали два индекса. Теперь мы также можем получать данные по индексу. Суть его по-прежнему осуществляется через API, который получал данные раньше, но исходный использовалkeyPath
Атрибуты преобразуются в атрибуты, указанные в индексе. Конкретные примеры следующие:
var request = window.indexedDB.open('test', 1);
request.onsuccess = function (event) {
var db = event.target.result;
var transaction = db.transaction(['table1'], 'readwrite');
var objectStore = transaction.objectStore('table1');
var index = objectStore.index('name');
// 第一种,get方法
index.get('a').onsuccess = function (event) {
console.log(event.target.result);
}
// 第二种,普通游标方法
index.openCursor().onsuccess = function (event) {
console.log('openCursor:', event.target.result.value);
}
// 第三种,键游标方法,该方法与第二种的差别为:普通游标带有value值表示获取的数据,而键游标没有
index.openKeyCursor().onsuccess = function (event) {
console.log('openKeyCursor:', event.target.result);
}
}
изменить данные
Когда нам нужно изменить данные в хранилище, мы можем использовать следующие API:
var objectStore = transaction.objectStore("customers");
var request = objectStore.put(data);
request.onsuccess = function (event) {
}
Примечание:put
Методы могут не только изменять существующие данные, но и добавлять новые данные в пространство для хранения.
удалить данные
Когда нам нужно удалить бесполезные данные, мы можем сделать следующее:
var objectStore = transaction.objectStore("customers");
var request = objectStore.delete(name);
request.onsuccess = function (event) {
}
Обработка исключений
Когда браузер выполняет следующие операции, в indexedDB может возникнуть исключение:
- Пользователь очищает кеш браузера
- Место для хранения превышает предельный размер
На этом этапе ошибка должна быть зафиксирована, и пользователю будет предложено. Эта глава не является основной темой данной статьи и будет здесь пропущена.
Расширенное уведомление
зависит от стоимости
Тип данных, который может принимать значение ключа
существуетIndexedDB
в паре ключ-значениеkey
Value может принимать следующие типы значений:
- number
- data
- string
- binary
- array
Подробности смотрите в документацииздесь.
Тип данных, который может принимать ключевой путь
когдаkey
Значение становится первичным ключом, т.е.keyPath
, его значение может быть только следующим:
- Blob
- File
- Array
- String
Примечание. В ключевом пути не должно быть пробелов..
Подробности смотрите в документацииздесь.
Тип данных, который может принимать значение
существуетIndexedDB
, value может принимать все типы значений в ECMA-262, такие как String, Date, ImageDate и т. д.
связанный с транзакцией
После того, как транзакция будет прервана, повлияет ли это на автоинкремент значения ключа?
IndexedDB
Если значение ключа не указано, будет использоваться самоувеличивающееся значение ключа. Если транзакция прерывается в середине, автоинкремент значения ключа начнется с ключа до начала прерванной транзакции.
связанных с безопасностью
IndexedDB
Также влияет браузерТа же политика происхожденияпределы.
связанные с пользователем
Очистить кэш
Когда пользователь очищает кеш браузера, он может очиститьIndexedDB
связанные данные.
разрешение на доступ
Некоторые браузеры, такие как режим конфиденциальности мобильной версии Safari, получают доступIndexedDB
, могут возникать исключения из-за отсутствия разрешений (как и LocalStorage), и требуется обработка исключений.
Суммировать
IndexedDB
Он играет незаменимую роль в локальном хранилище и является альтернативой реляционным базам данных.web sql
продукты, способные хранить большие объемы данных. Во многих сценариях, где требуется автономное хранилище, оно может оказать нам эффективную поддержку.
но,IndexedDB
В процессе использования все же необходимо избегать некоторых проблем, которые могут возникнуть, или иметь определенную отказоустойчивость к неблагоприятным последствиям, которые могут быть вызваны. Это не окажет существенного влияния на приложение.