Мастер-классы по Javascript, AJAX/COMET, jQuery Узнать больше...
Javascript.RU

Хранение данных на клиенте. DOM Storage и его аналоги.

По мере того как web-странички превращаются в AJAX-приложения, им требуются все новые возможности.

Сложные выборки элементов DOM обеспечиваются некоторыми браузерами и почти всеми распространенными Javascript-фреймворками.

Кросс-доменные HTTP-запросы находят поддержку в стандартах и реализуются в новейших браузерах, включая Internet Explorer 8.

В этой статье пойдет речь о средствах для хранения большого количества данных на клиенте, в браузере, которые доступны уже сейчас.

В частности, Internet Explorer 5+, Firefox 2+, Safari 2+ не требуют для этого дополнительных плагинов и Flash.

Почти во всех браузерах есть поддержка cookies.

На протяжении долгого времени cookies были единственным кросс-браузерным способом сохранить данные, которые будут доступны после перезагрузки страницы.

Однако у cookie есть две важных особенности:

  1. Не более 2 килобайт данных
  2. Данные идут на сервер при каждом HTTP-запросе

Средства хранения на клиенте предусматривают сотни килобайт и мегабайты данных, и не отсылают их на сервер при каждом HTTP-запросе.

А cookie можно продолжать использовать, например, для хранения сессии.

Firefox реализует стандарт хранения "Client-side session and persistent storage of name/value pairs", предложенный в спецификации HTML 5.

Для постоянного хранения данных в нем используется объект window.globalStorage[домен], операции над которым можно производить точно так же, как над обычным javascript-объектом. При уходе с сайта и даже закрытии браузера globalStorage не меняется, так что все его свойства можно прочитать обратно.

Например:

storage = globalStorage[document.domain]

// записать значение
storage['userName'] = 'Vasya'
// прочитать значение
alert(storage['userName'])
// удалить значение
delete storage['userName']

// получить все значения
for(var name in storage) {
    alert(name + ':' + storage[name])
}

При чтении/записи на элементе body инициируется всплывающее событие storage.

Поймать его можно, например, таким обработчиком:

window.addEventListener('storage', function(event) { ... })

Стандарт HTML 5 все еще в процессе развития. В старой редакции прочитанные значения имели тип StorageItem.

Версия Firefox 2.0.0.13 возвращает при чтении объект именно этого типа.

Из текущей редакции StorageItem убран. В ней возвращаемые хранилищем значения имеют более простой тип DOMString.

..А пока эти изменения не учтены разработчиками, рекомендуется преобразовать значения к String явным образом.

Например:

var test = "12345"
storage.test = test // сохранить -> String
test = storage.test // прочитать <- StorageItem

alert(test.length) // undefined, это же не строка
alert(test.constructor) // StorageItem

test = String(test) // сделали строку. Теперь все ок.

Ограничения на данные: ключи и значения - только строки.

Размер: 5MB на домен.

Ограничения безопасности - точно такие же, как на cookie.
Данные в globalStorage['site.ru'] можно сохранить только на самом site.ru, а прочитать - на blog.site.ru, но не на otherhost.ru.

Internet Explorer 8 реализует DOM Storage, в то время как версии начиная от 5й поддерживают собственный интерфейс: userData behavior.

Он работает посредством выделенного DOM-элемента, которому назначается behavior userData. В этот элемент загружается нужное пространство имен, и данные становятся доступны через атрибуты.

<span id="storageElement"></span>
<script>    
    storage = document.getElementById('storageElement')
    if (!storage.addBehavior) {
        alert("userData not available.")
    } else {
        // поставить userData behavior
        storage.addBehavior("#default#userData")
        // загрузить пространство имен
        storage.load("namespace")    
    }
</script>

После инициализации можно работать с данными. Для записи изменений используется метод save.

function put(key, value) { // записать значение
    storage.setAttribute(key, value)
    storage.save("namespace")
}

function get(key) { // получить значение
    return storage.getAttribute(key)
}

function remove(key) { // удалить значение
    storage.removeAttribute(key)
    storage.save("namespace")
}

Как это часто бывает с Internet Explorer, некоторые операции делаются неочевидным образом.

Так, например, получить все сохраненные данные из storage.attributes нельзя. Там хранятся только атрибуты самого HTML-элемента.

Данные же хранятся в свойстве storage.XMLDocument.documentElement.attributes.

Например, следующий код создает список вида ключ:значение.

var list = []
var attrs = storage.XMLDocument.documentElement.attributes
        
for(var i=0; i<attrs.length; i++) {
    list.push(attrs[i].name+':'+attrs[i].value);
}

В отличие от DOM Storage, можно задать атрибут expires. Он устанавливается на уровне всего элемента и действует на все хранящиеся данные. Очистка данных происходит при вызове load.

var time = new Date(); // Start Time
time.setMinutes(time.getMinutes() + 1)
storage.expires = time.toUTCString();

Ключи и значения - только строки.

Способ работает при всех уровнях безопасности, кроме "Высокого".
При этом для сайтов в зоне Internet объем ограничен 128K на страницу и 1024K на домен, для локальных и интранет - лимит увеличен.

Ограничения безопасности - та же директория, тот же домен и протокол.

Дальше всех в поддержке стандарта хранения пошли разработчики WebKit.
В Safari реализовано локальное хранение в базе данных SQLite.

Набор операций включает в себя CREATE TABLE, INSERT, SELECT, REPLACE, индексы и многое другое, с рядом ограничений безопасности (например, нет LOAD DATA INFILE).

В отличие от DOM Storage и userData, этот интерфейс асинхронный. Все функции запросов к базе данных принимают в качестве аргументов две функции: callback - для обработки результатов и errback - для обработки ошибок.

Когда запрос завершается, вызывается один из этих обработчиков.

Продемонстрируем это на тестовой базе.

db = openDatabase("Test", "1.0", "Webkit Storage Example")

db.transaction(function(tx) {
  tx.executeSql(
      "CREATE TABLE IF NOT EXISTS test (key TEXT, value TEXT, unique(key))", 
      [], 
      function(tx, result) { alert("Success!") },
      function(tx, error) { alert("Failure: "+error.message }
  )
})

Сложновато с первого взгляда?

db.transaction создает транзакцию и передает ее функции-аргументу.

Код внутри function(tx) выполняется в одной транзакции.

Вызов tx.executeSql принимает аргументы:

  1. Запрос
  2. Аргументы подстановки
  3. Обработчик результата
  4. Обработчик ошибки

Следующий пример демонстрирует обработку запроса.

db.transaction(function(tx) {
  tx.executeSql("SELECT value FROM test WHERE key=?", [key],
    function(tx,result) {
      alert("Количество результатов: "+result.rows.length)
      alert("Поле value первого результата: "+ result.rows.item(0).value)
   },
   function(tx, error) { alert("Error!") }
  )
})

Стандарт SQL-хранения также включает в себя поддержку версий схемы, указание размера базы данных в openDatabase и многое другое. Может существовать только одна версия схемы одновременно.

База существует только в рамках домена(полного домена, origin), на котором была создана. Поддомен не имеет доступа к базе домена.

На момент написания статьи разработчики WebKit планировали поддержку DOM Storage, но в nightly build ее не было.

На момент написания статьи Opera 9.5 (beta) не поддерживает ни DOM Storage ни Database Storage.

С другой стороны, разработчики планируют эту поддержку включить.

Там, где нет DOM Storage, для offline-хранения используют flash-интерфейс SharedObject. Он позволяет хранить самые разные объекты средствами Adobe Flash.

Пример ActionScript для работы с SharedObject:

// создать/получить namespace storage
storage = SharedObject.getLocal("storage"); 

// записать данные name => Вася
storage.data.name="Вася";

// сохранить объект
storage.flush()

// перечислить свойства объекта
for (var name in storage.data) {
    trace(name + ":" + storage.data[name])
}

Чтобы работать с этим хранилищем из javascript, нужен способ коммуникации JS<->Flash.

В старых версиях Flash вызвать javascript можно через getURL('javascript:...').

Передать значение во Flash можно установкой переменной flash-объекту. Эту переменную flash-ролик может считывать каждый кадр и предпринимать соответствующие действия.

Во Flash 8+ появился интерфейс ExternalInterface, который позволяет как указывать AS-функцию для приема данных из JS, так и напрямую вызывать JS-метод.

Открыть рабочий пример передачи значения Flash <-> JS.

Код примера в ActionScript:

import flash.external.*;

// установить местную функцию recieveFromJS для приема данных
// от javascript-функции sendFromJS
ExternalInterface.addCallback("sendFromJS", null, recieveFromJS);

// Эта функция будет реагировать на sendFromJS
function recieveFromJS(text) {
    _root.theText.text = text; // .. и устанавливать текст в окошке
}

// Это действие, наоборот, отправляет данные в JS.
_root.button.onRelease = function() {
    // вызвать javascript-функцию recieveFromFlash
    ExternalInterface.call("recieveFromFlash", _root.theText.text);
    _root.theText.text = "";
}

Код примера в JS:

function recieveFromFlash(Txt) {
	document.getElementById('text').value = Txt;
}

function sendFromJS() {
    var value = document.getElementById('text').value
    
    var movie = (navigator.appName.indexOf("Microsoft")!=-1 ? window : document)["BridgeMovie"]    
    
    movie.sendFromJS(value);    

    document.getElementById('text').value = ''
}

Скачать исходники

Документация на ExternalInterface

Доступ к SharedObject ограничен роликами с того же домена.

Это принципиально отличается от Javascript, в котором доступ определяется адресом страницы а не скрипта, и делает возможным разного рода кросс-доменные трюки.

Ограничение по умолчанию на размер данных - в районе 100Kb, пользователь может уменьшить или увеличить его в специальном Flash-интерфейсе, который открывается при вызове ActionScript:

System.showSettings(1);

Во-первых, надо иметь Flash. Хранилище доступно только после инициализации Flash-ролика.

Много ошибок в различных версиях Flash затрагивают ExternalInterface, включая повреждение данных во время передачи JS->Flash.

Проще всего узнать о них:

Много работы над обходом багов провел Brad Neuberg для flash-хранилища в dojo:

  • dojox/storage - различные хранилища, включая flash
  • dojox/flash - кроссбраузерная js/flash коммуникация, необходима для dojox/storage

DOM Storage и аналогичные системы хранения - важный шаг к offline-работе web-приложений.

В браузере Opera все еще приходится использовать нужен Flash Shared Object, а Firefox, Internet Explorer, Safari реализуют хранилище без дополительных плагинов и Flash<->JS коммуникации.

Интересным сектором для интеграции могут быть CRM-системы с нестабильным интернетом. Например, мини-точка продаж, в которой основная работа идет через интеренет, но хранение текущих договоров дублируется на рабочей машине.

Offline-хранилище может быть использовано для сохранения сложных состояний интерфейса - размеров окон, контрольных элементов и т.п.

Все изменения интерфейса посетителем мгновенно сохранятся в DOM Storage и восстановятся при следующем заходе на страницу без дополнительных механизмов сохранения интерфейса на сервере.

При этом объем хранимого состояния интерфейса вряд ли превысит ограничение в сотни килобайт. Идеальный объект для DOM Storage, не правда ли ?..


Автор: vashurin, дата: 6 мая, 2008 - 09:05
#permalink

Автор: Илья Кантор, дата: 25 мая, 2008 - 18:21
#permalink

dojo svn, похоже, закрылся или переместился. Поменял ссылку на путь внутри dojo.


Автор: Гость (не зарегистрирован), дата: 19 июня, 2008 - 14:19
#permalink

Данные, в globalStorage['site.ru'] можно сохранить только на самом site.ru, а прочитать - на blog.site.ru...

В FF3 данную возможность убрали по соображениям безопастности.


Автор: Гость (не зарегистрирован), дата: 29 августа, 2008 - 16:15
#permalink

у меня в FF3 не сработал пример с хранилищем


Автор: Евгений (не зарегистрирован), дата: 15 октября, 2008 - 13:10
#permalink

Мне на запись

var storage = globalStorage[document.domain];

Mozilla/5.0 (Windows; U; Windows NT 5.1; ru; rv:1.8.1.17) Gecko/20080829 Firefox/2.0.0.17 выдал ошибку Security error" code: "1000


Автор: Илья Кантор, дата: 15 октября, 2008 - 15:40
#permalink

Примеры на сайте работают в FF3 и FF2 у меня (FF2,3 под Linux).


Автор: Алексей (не зарегистрирован), дата: 30 октября, 2008 - 08:48
#permalink

storage.load("namespace") - на это строке выкидывает ошибку "разрешение отклонено"


Автор: Дмитрий) (не зарегистрирован), дата: 13 ноября, 2008 - 17:08
#permalink

DOM Storage и аналогичные системы хранения - важный шаг к offline-работе web-приложений.


Автор: oxyum, дата: 29 ноября, 2008 - 01:41
#permalink

А почему бы в этом разделе не описать уже весьма развитое решение от Google?

Зовётся оно Google Gears, ставится плагином на разные браузеры, внутри основано на SQLite.


Автор: Гость (не зарегистрирован), дата: 23 февраля, 2009 - 19:09
#permalink

опишите, пожалуйста:)


Автор: Яростный Меч, дата: 10 июня, 2010 - 17:22
#permalink

В таком случае для работы веб-приложения потребуется, чтобы этот плагин стоял у пользователей, большинство из которых вообще не в курсе, что, к чему и для чего.


Автор: yak2689 (не зарегистрирован), дата: 3 декабря, 2008 - 15:48
#permalink

Странная эта штука - DOM Storage - для одного НТML-документа
в Mozilla своя память, в IE-своя и, находясь на одной машине, друг друга не видят.
Можно ли в среде НТML/JS писать/читать определенный
(текстовый) файл, как это делается, например, в Си (Си++)?
Вообще, как подать параметры/данные из одного НТML-документа
в другой? Можно ли клиент-скрипт написать на Си (Си++)?


Автор: macint0sh, дата: 27 сентября, 2009 - 02:41
#permalink

Уж если я правильно понял, то суть проблемы описываемой в статье, это хранение данных(объемом большим чем позволенный в cookies) на клиенте, ну и так сказать наиболее удобный для этого подход.
Как выше говорилось, почему бы не использовать google gears?
Ну и насколько я знаю, есть вот такое чудо, TaffyDB, которое решает все проблемы кроссбраузерности при этом предоставляя удобный интерфейс для работы с данными.
Возможно неправильно понял суть, поправьте...


Автор: Snipe, дата: 1 октября, 2009 - 11:57
#permalink

А какие ограничения на название ключа? Например () нельзя использовать, а что еще?


Автор: Гость (не зарегистрирован), дата: 19 ноября, 2009 - 21:53
#permalink

Ещё ссылки на некоторые библиотеки для работы с локальными хранилищами:
YUI 2: Storage Utility
The Dojo Offline Toolkit
iLocalStorage
jStore


Автор: творожок (не зарегистрирован), дата: 1 декабря, 2009 - 01:30
#permalink

или я дурак или лыжи не едут. нет у меня в FireFox в версии 3.0.13 никакого globalStorage. В итоге кроме ошибки ничего данные скрипты не дают.


Автор: творожок (не зарегистрирован), дата: 1 декабря, 2009 - 02:26
#permalink

А в ишаке работает!


Автор: Lexi, дата: 28 декабря, 2009 - 16:05
#permalink

К альфе Opera 10.50 таки прикрутили DOM Storage и Database Storage.


Автор: Илья Кантор, дата: 28 декабря, 2009 - 19:31
#permalink

Ну и новость!


Автор: мамбет (не зарегистрирован), дата: 16 февраля, 2010 - 23:12
#permalink

Братан, тебе надо переписать статью.

Выкинуть все из статьи и описать window.localStorage.

Так как эта штука поддерживается FF 3.6, Opera 10.5, IE 8.0, Сhrome 4.0, Safari 4.0. Сейчас проверил, работает window.localStorage во всех этих браузерах. А для старых IE 6-7 оставить твое описание userData.


Автор: Гость (не зарегистрирован), дата: 8 марта, 2010 - 21:01
#permalink

респект, мамбет!
psкак однако полезно читать каменты (иногда)


Автор: Гость (не зарегистрирован), дата: 19 марта, 2010 - 11:45
#permalink

>все еще приходится использовать нужен
поправьте или не поправляйте. как хотите.


Автор: Хулиос (не зарегистрирован), дата: 26 апреля, 2010 - 20:54
#permalink

Здравствуйте, уважаемые!

Я намереваюсь использовать DOM Storage (Firefox) для хранения небольшой базы данных своего скрипта для Greasemonkey.

Задался вопросом: "А где же все эти данные находятся физически?"

У меня будет около 300-от записей (в каждой от 5-ти до 10-ти "ячеек").
Хотелось бы обнаружить этот файлик и сделать его резервную копию.

Это возможно?


Автор: Хулиос (не зарегистрирован), дата: 26 апреля, 2010 - 22:34
#permalink

Нашел:
C:\Documents and Settings\Администратор\Application Data\Mozilla\Firefox\Profiles\s6v6hjub.default\webappsstore.sqlite


Отправить комментарий

Приветствуются комментарии:
  • Полезные.
  • Дополняющие прочитанное.
  • Вопросы по прочитанному. Именно по прочитанному, чтобы ответ на него помог другим разобраться в предмете статьи. Другие вопросы могут быть удалены.
    Для остальных вопросов и обсуждений есть форум.
P.S. Лучшее "спасибо" - не комментарий, как все здорово, а рекомендация или ссылка на статью.
Содержание этого поля является приватным и не предназначено к показу.
  • Адреса страниц и электронной почты автоматически преобразуются в ссылки.
  • Разрешены HTML-таги: <strike> <a> <em> <strong> <cite> <code> <ul> <ol> <li> <dl> <dt> <dd> <u> <i> <b> <pre> <img> <abbr> <blockquote> <h1> <h2> <h3> <h4> <h5> <p> <div> <span> <sub> <sup>
  • Строки и параграфы переносятся автоматически.
  • Текстовые смайлы будут заменены на графические.

Подробнее о форматировании

CAPTCHA
Антиспам
1 + 2 =
Введите результат. Например, для 1+3, введите 4.
 
Новости

Открылась регистрация на мастер-классы по профессиональному Javascript, AJAX/COMET, jQuery в городах:

  • Ярославль (24-25 сентября)
  • Новосибирск (3-4 октября)
  • Казань (9-10 октября)
  • Минск (16-17 октября)
  • Днепропетровск (23-24 октября)
  • Одесса (30-31 октября)
  • Самара (13-14 ноября)

Более подробно - на странице мастер-классов.

Если вас интересует другой город - посмотрите здесь, выбрав "Другие города".

Текущий раздел
Поиск по сайту
Содержание

Учебник javascript

Основные элементы языка

Сундучок с инструментами

Интерфейсы

Все об AJAX

Оптимизация

Разное

Дерево всех статей

Статьи и мероприятия

Будьте в курсе наших последних новостей!

Последние обсуждения на форуме
Forum
Последние комментарии