d.close
puts d['123'] # RuntimeError: закрытый DBM-файл.
e = DBM.open('data')
e['123'] # 'toodle-oo!'
w=e.to_hash # {'123'=>'toodle-oo!'}
e.close
e['123'] # RuntimeError: закрытый DBM-файл.
w['123'] # 'toodle-oo!
Интерфейс к DBM реализован в виде одного класса, к которому подмешан модуль Enumerable
. Два метода класса (синонимы) new
и open
являются синглетами, то есть в любой момент времени можно иметь только один объект DBM, связанный с данным файлом.
q=DBM.new('data.dbm') #
f=DBM.open('data.dbm') # Errno::EWOULDBLOCK:
# Try again - 'data.dbm'
Всего есть 34 метода экземпляра, многие из которых являются синонимами или аналогичны методам хэша. Почти все операции с настоящим хэшем применимы и к объекту dbm
.
Метод to_hash
создает представление файла в виде хэша в памяти, а метод close
закрывает связь с файлом. Остальные методы по большей части аналогичны методам хэшам, однако дополнительно есть методы rehash
, sort
, default
, default=
. Метод to_s
возвращает строковое представление идентификатора объекта.
10.3. Библиотека KirbyBase
KirbyBase — небольшая библиотека, с которой должен освоиться каждый программист на Ruby. В настоящее время она не входит в стандартный дистрибутив, а если бы входила, то была бы еще полезнее.
KirbyBase — плод трудов Джейми Криббса (Jamey Cribbs), названный, к слову, в честь его собаки. Во многих отношениях это полноценная база данных, но есть причины, по которым мы рассматриваем ее здесь, а не вместе с MySQL и Oracle.
Во-первых, это не автономное приложение. Это библиотека для Ruby, и без Ruby ее использовать нельзя. Во-вторых, она вообще не знает, что такое язык SQL. Если вам без SQL не обойтись, то эта библиотека не для вас. В-третьих, если приложение достаточно сложное, то функциональных возможностей и быстродействия KirbyBase может не хватить.
Но несмотря на все это, есть немало причин любить KirbyBase. Это написанная целиком на Ruby библиотека, состоящая из единственного файла, которую не нужно ни устанавливать, ни конфигурировать. Она работает на всех платформах, и созданные с ее помощью файлы можно переносить с одной платформы на другую. Это «настоящая» база данных в том смысле, что данные не загружаются целиком в память.
Библиотекой легко пользоваться, а ее интерфейс выдержан в духе Ruby с легким налетом DBI. В общем, база данных соответствует каталогу, а каждая таблица — одному файлу. Формат данных в таблицах таков, что человек может их читать (и редактировать). Дополнительно таблицы можно зашифровать — но только для того, чтобы затруднить редактирование. База знает об объектах Ruby; допускается их хранение и извлечение без потери информации.
Наконец, благодаря интерфейсу dRuby библиотека может работать в распределенном режиме. К данным, хранящимся в KirbyBase, можно с одинаковым успехом обращаться как с локальной, так и с удаленной машины.
Чтобы открыть базу данных, нужно сначала указать, является ли она локальной. Следующие два параметра обычно равны nil
, а четвертый указывает каталог, в котором будут храниться файлы с данными (по умолчанию это текущий каталог).
Чтобы создать таблицу, вызывается метод create_table
объекта, представляющего базу данных; ему передается имя таблицы (объект Symbol
); имя файла на диске образуется из этого имени. Затем передается последовательность пар символов, описывающих имена и типы полей.
require 'kirbybase'
db = KirbyBase.new(:local, nil, nil, 'mydata')
books = db.create_table(:books, # Имя таблицы.
:title, :String, # Поле, тип, ...
:author, :String)
В текущей версии KirbyBase распознает следующие типы полей: String
, Integer
, Float
, Boolean
, Time
, Date
, DateTime
, Memo
, Blob
и YAML
. К тому моменту, когда вы будете читать эту главу, возможно, появятся и новые типы.
Для вставки записи в таблицу применяется метод insert
. Ему можно передать список значений, хэш или любой объект, отвечающий на заданные имена полей.
books.insert('The Case for Mars','Robert Zubrin')
books.insert(:title => 'Democracy in America',
:author => 'Alexis de Tocqueville')
Book = Struct.new(:title, :author)
book = Book.new('The Ruby Way','Hal Fulton')
books.insert(book)
В любом случае метод insert
возвращает идентификатор строки, соответствующей новой записи (вы можете использовать его или игнорировать). Это «скрытое» автоинкрементное поле, присутствующее в каждой записи любой таблицы. Для выборки записей служит метод select
. Без параметров он выбирает все поля всех записей таблицы. Набор полей можно ограничить, передав в качестве параметров символы. Если задан блок, то он определяет, какие записи отбирать (примерно так же, как работает метод find_all
для массивов).
list1 = people.select # Все люди, все поля.
list2 = people.select(:name,:age) # Все люди, только имя и возраст.
list3 = people.select(:name) {|x| x.age >= 18 && x.age < 30 }
# Имена всех людей от 18 до 30 лет.
В блоке допустимы любые операции. Это означает, например, что можно формулировать запрос с помощью регулярных выражений (в отличие от типичной SQL-базы).
Результирующий набор, возвращаемый KirbyBase, можно сортировать по нескольким ключам в порядке возрастания или убывания. Для сортировки по убыванию перед именем ключа ставится минус. (Это работает, потому что в класс Symbol
добавлен метод, соответствующий унарному минусу.)
sorted = people.select.sort(:name,-:age)