Книга: Программирование на языке Ruby

10.2.1. Простой маршалинг

10.2.1. Простой маршалинг

Часто бывает необходимо создать объект и сохранить его для последующего использования. В Ruby есть рудиментарная поддержка для обеспечения устойчивости объекта или маршалинга. Модуль Marshal позволяет сериализовать и десериализовать объекты.

# Массив элементов [composer, work, minutes]
works = [["Leonard Bernstein","Overture to Candide",11],
["Aaron Copland","Symphony No. 3",45],
["Jean Sibelius","Finlandia",20]]
# Мы хотим сохранить его для последующего использования...
File.open("store","w") do |file|
 Marshal.dump(works,file)
end
# Намного позже...
File.open("store") do |file|
 works = Marshal.load(file)
end

Недостаток такого подхода заключается в том, что не все объекты можно сохранить. Для объектов, включающих другие объекты низкого уровня, маршалинг невозможен. К числу таких низкоуровневых объектов относятся, в частности, IO, Proc и Binding. Нельзя также сериализовать синглетные объекты, анонимные классы и модули.

Метод Marshal.dump можно вызывать еще двумя способами. Если он вызывается с одним параметром, то возвращает данные в виде строки, в которой первые два байта — это номер старшей и младшей версии.

s = Marshal.dump(works)
p s[0] # 4
p s[1] # 8

Обычно попытка загрузить такие данные оказывается успешной только в случае, если номера старших версий совпадают и номер младшей версии данных не больше младшей версии метода. Но если при вызове интерпретатора Ruby задан флаг «болтливости» (verbose или v), то версии должны совпадать точно. Эти номера версий не связаны с номерами версий Ruby.

Третий параметр limit (целое число) имеет смысл, только если сериализуемый объект содержит вложенные объекты. Если он задан, то интерпретируется методом Marshal.dump как максимальная глубина обхода объекта. Если уровень вложенности меньше указанного порога, то объект сериализуется без ошибок; в противном случае возбуждается исключение ArgumentError. Проще пояснить это на примере:

File.open("store","w") do |file|
 arr = []
 Marshal.dump(arr,file,0) # Внутри 'dump': превышена пороговая глубина.
                          # (ArgumentError)
 Marshal.dump(arr,file,1)
 arr = [1, 2, 3]
 Marshal.dump(arr,file,1) # Внутри 'dump': превышена пороговая глубина.
                          # (ArgumentError)
 Marshal.dump(arr,file,2) arr = [1, [2], 3]
 Marshal.dump(arr,file,2) # Внутри 'dump': превышена пороговая глубина.
                          # (ArgumentError)
 Marshal.dump(arr,file,3)
end
File.open("store") do |file|
 p Marshal.load(file) # [ ]
 p Marshal.load(file) # [1, 2, 3]
 p Marshal.load(file) # arr = [1, [2], 3]
end

По умолчанию третий параметр равен 1. Отрицательное значение означает, что глубина вложенности не проверяется.

Оглавление книги


Генерация: 1.365. Запросов К БД/Cache: 3 / 0
поделиться
Вверх Вниз