Курсоры
НЕТ ВОЙНЕ |
24 февраля 2022 года российское руководство во главе с Владимиром Путиным развязало агрессивную войну против Украины. В глазах всего мира это военное преступление совершено от лица всей страны, всех россиян. Будучи гражданами Российской Федерации, мы против своей воли оказались ответственными за нарушение международного права, военное вторжение и массовую гибель людей. Чудовищность совершенного преступления не оставляет возможности промолчать или ограничиться пассивным несогласием. Мы убеждены в абсолютной ценности человеческой жизни, в незыблемости прав и свобод личности. Режим Путина — угроза этим ценностям. Наша задача — обьединить все силы для сопротивления ей. Эту войну начали не россияне, а обезумевший диктатор. И наш гражданский долг — сделать всё, чтобы её остановить. Антивоенный комитет России |
Распространяйте правду о текущих событиях, оберегайте от пропаганды своих друзей и близких. Изменение общественного восприятия войны - ключ к её завершению. |
meduza.io, Популярная политика, Новая газета, zona.media, Майкл Наки. |
Определение: |
Курсор — способ перебора результатов запроса, позволяющий достать из результата запроса множество строк. |
Курсор можно рассматривать как итератор, специфичный для баз данных.
Запрос исполнился, мы получили какие-то результаты, и теперь собираемся достать множество строк. Все, что обсуждалось до этого (set и select into) позволяло доставать только один результат, а из курсора их можно достать множество.
Содержание
Объявление курсора
Синтаксис
declare имя -- объявление курсора с именем {sensitive|insensitive|asensitive} -- чувствительность курсора [[no] scroll] -- возможность ходить произвольным образом по данным вперед и назад (default - no scroll) cursor select... -- запрос данных, по которым будет ходить курсор [for {read only|update [of столбцы]}]; -- права на чтение (дефолтное право) или на обновление (всей таблицы (в частности, этот вариант подходит для удаления) или выбранных столбцов)
Чувствительность курсора
Данные из курсора обрабатываются по одной строке (способа получить более одной строки за один раз нет). Мы напишем цикл, в рамках которого будут выбираться данные из курсора, и возможно будут меняться данные в нашей базе данных. Получаем, что во время таких итераций можно поменять данные самой таблицы, по которой построен текущий курсор. Представим ситуацию, что мы в сделанном курсоре в явной форме написали update на ту таблицу или таблицы, по которым сделан курсор. Что должно произойти в таком случае? В зависимости от желаемого поведения можно выбрать разную чувствительность курсора.
- - курсор, чувствительный к изменениям. Измененные данные можно будет увидеть в курсоре, если эти данные еще не были пройдены. Иначе, если изменённые данные были просмотрены в прошлом и у нас не scrollable курсор, то они не обновятся. В некотором смысле это итерация по реальным данным, динамическая.
- - курсор фиксированных данных. Данные зафиксированы на момент открытия курсора. Никакие изменения данных не будут отображены через курсор. В некотором смысле это итерация по временной копии таблицы.
- - курсор, наиболее предпочтительный для данной СУБД. Связано с различными способами реализациями СУБД: для одних легче реализуем sensitive cursor, а для других - insensitive. Существует два подхода к реализации СУБД: snapshot-based и lock-based, для них либо sensitive, либо insensitive варианты предпочтительны. Ни одна из опций не является дешевой по умолчанию, поэтому и есть значение asensitive: сама СУБД будет решать, какую чувствительность курсора использовать. Для lock-based проще сделать курсор, чувствительный к изменениям, так как для insensitive потребовалось бы дополнительное копирование. Для snapshot-based предпочтительнее insensitive, так как не потребуется специальным образом учитывать обновления.
Если данные во время использования курсора меняться не будут, то чувствительность курсора не важна.
Scrollable | No scroll курсоры
- - по такому курсору можно ходить только по одной записи вперед, как из обычного итератора в Java.
- - по такому курсору можно ходить не только по одной записи вперед, но и вперед, назад, в начало или конец. Такой курсор больше похож на итератор из языка С++.
Примеры
Взятие курсора на чтение для всех студентов, упорядоченных по идентификатору:
declare SCursor cursor select * from Students order by SId;
Взятие курсора с правом на изменение поля SName для студентов группы M3439 в произвольном порядке:
declare SCursor cursor select * from Students where GId = 'M3439' for update of SName;
Открытие и закрытие
Объявление курсора говорит о намерении получения данных. Реальная выборка в момент объявления не производится. Она происходит в момент открытия курсора (выполнения действия open cursor). Если курсор insensitive, то в момент открытия фиксируются данные, которые попадут в курсор.
С курсором ассоциируются множество ресурсов, поэтому их необходимо закрывать. При выходе из блока, в котором был объявлен курсор, он будет автоматически закрыт. Это чем-то напоминает автоматическую сборку ресурсов в sql. При повторном открытии курсора произойдет повторная инициализация, влекущая новое прочтение данных для insensitive курсора.
Синтаксис
open курсор; close курсор;
Получение данных
Для получения данных из курсора есть конструкция fetch.
Синтаксис
fetch [next|prior|last|first|absolute n|relative n] курсор into переменные;
Описание команд
- - получение следующей записи (доступно по умолчанию, все команды ниже доступны только для scroll курсора)
- - предыдущая запись
- - последняя запись
- - первая запись
- - получить запись с номером n от начала
- - запись с номером n, нумерация с текущей позиции (n может быть как положительным (движение в сторону конца), так и отрицательным (в этом случае движение назад))
Отсутствие данных
Если данные закончились, либо при помощи команд выше был осуществлен невалидный переход, то случится событие not found.
Пример
В данном примере из курсора для студентов мы получим идентификатор, имя и фамилию того студента, на которого сейчас указывает курсор.
fetch SCursor into Sid, FirstName, LastName;
Для получения всех данных из курсора, можно создать цикл, предварительно создав обработчик ошибок на not found снаружи цикла, в котором выйдем из цикла, когда все элементы будут обработаны. Состояние sqlstate 02xxx не является исключительной ситуацией, а просто предупреждение от СУБД, которое зачастую игнорируется. Вопрос производительности зависит от реализации СУБД.
Позиционированные изменения
Изменение или удаление ровно одной строчки, которую последней выдал курсор (только в update
и delete
).
Синтаксис
update таблица set что where current of курсор; -- изменение ровно одной строки, которую последней выдал курсор delete from таблица where current of курсор; -- удаление последней выданной строки
Пример
if FirstName = 'Иван' then delete from Students where current of SCursor; end if;
В данном примере мы удалим текущего студента, если его имя Иван.
Пример применения курсора
create procedure updateNameStat(name) begin declare cnt int default 0; declare exit handler for sqlstate not found begin insert into Stats (Name, Cnt) values (name, cnt); end; declare SCursor cursor select FirstName from Students; open SCursor; loop declare sname varchar(20); fetch SCursor into sname; if name = sname then set cnt = cnt + 1; end if; end loop; end;
Пояснения
В данном примере объявлен счетчик cnt
, по умолчанию проинициализированный 0, курсор SCursor
, перебирающий значение столбца FirstName
из таблицы Students
. Далее в бесконечном цикле (так как мы не имеет возможности написать условие выхода из цикла):
- определяем переменную
sname
, в которой будем хранить имя студента - достаем следующую запись в эту переменную (напоминание:
fetch
по умолчанию - этоfetch next
) - проверяем, если текущее имя
sname
имя соответствует выделенному имениname
, то увеличиваем счетчик.
Повторяем данные операции до тех пор, пока у нас не случится not found
, при возникновении которого в обработчике not found
обновим данные статистики и (по определению обработчика в sql) выйдем из блока, в котором был объявлен handler
- то есть в данном случае выйдем из блока процедуры.
Обработчик not found
может быть только один. Если бы у нас было несколько курсоров, и мы бы захотели знать в обработчике, в каком из курсоров закончились данные, то предварительно вели бы записи в переменные, из каких курсоров планируем достать данные. И в обработчике проверили бы значение переменных.
Данные пример написан исключительно для демонстрации работы с курсорами. В реальной жизни подобное поведение может быть достигнут проще с использованием select
и count
.
Примечания
Некоторые базы данных позволяют проверить, есть ли в курсоре еще данные. В этих случаях пишется цикл while
с вызовом метода проверки. Однако стандартом наличие этой проверки не предусмотрено.
Особенности курсоров
Курсоры, описанные в данной статье, являются внутренними. Они существуют внутри базы данных, используются, в том числе для взаимодействия между различными частями базы данных при планировании.
Внешние курсоры
Результат запроса к базе данных может быть слишком велик и стать виной нехватки памяти как на стороне клиента, так и на стороны сервера, увеличить нагрузку на сеть, вызывать и другие проблемы. Поэтому наружу данные выдаются в виде внешнего курсора. После исполнения команды select выдается внешний курсор на данные, аналогично внешний курсор выдается при возврате результата из хранимой процедуры.
Размер выборки
Может быть указан для внешнего курсора каким-либо образом (в количестве записей или в объеме данных, который выдается за один раз).
При выдаче курсора приходит метаинформация о курсоре и первая порция данных, размер которой мы указали. Для получения последующих порций данных необходимо обращаться к серверу, что будет медленнее, чем получение первой порции. В конце сервер сообщит о том, что данные закончились, автоматически освободит ресурсы.
Возникают альтернативы:
- Малый размер выборки — много повторных обращений к серверу базы данных, а так как обычно такое обращение происходит по сети, то взаимодействие не слишком быстрое.
- Большой размер выборки — требуется много памяти, как со стороны сервера для генерации данных, так и со стороны прикладного приложения.
Выдача курсора приводит к расходу ресурсов сервера (память, возможно дисковая память, блокировки и так далее). Поэтому внешний курсор стоит закрыть как можно раньше.
Однако чтобы не получить слишком много данных по запросу, они выдаются в виде внешнего курсора.
Литература
- Дейт К. Введение в системы баз данных (главы 4 и 17)
- Уидом Д., Ульман Д. Основы реляционных баз данных (раздел 7.4)
- Gulutzan P., Pelzer T. SQL-99 complete, really