9.16. Функции для работы с последовательностями
В этом разделе описаны функции для работы с объектами, представляющими последовательности. Такие объекты (также называемыми генераторами последовательностей или просто последовательностями) являются специальными таблицами из одной строки и создаются командой CREATE SEQUENCE. Используются они обычно для получения уникальных идентификаторов строк таблицы. Функции, перечисленные в Таблице 9.44, предоставляют простые и безопасные для параллельного использования методы получения очередных значений таких последовательностей.
Таблица 9.44. Функции для работы с последовательностями
Функция | Тип результата | Описание |
---|---|---|
| bigint | Выдаёт значение заданной последовательности, которое было возвращено при последнем вызове функции nextval |
| bigint | Выдаёт значение любой последовательности, которое было возвращено при последнем вызове функции nextval |
| bigint | Продвигает последовательность к следующему значению и возвращает его |
| bigint | Устанавливает текущее значение последовательности |
| bigint | Устанавливает текущее значение последовательности и флаг is_called , указывающий на то, что это значение использовалось |
Последовательность, к которой будет обращаться одна из этих функций, определяется аргументом regclass
, задающим просто OID последовательности в системном каталоге pg_class
. Вычислять этот OID вручную не нужно, так как процедура ввода данных regclass
автоматически выполнит эту работу за вас. Просто запишите имя последовательности в апострофах, чтобы оно выглядело как строковая константа. Для совместимости с обычными именами SQL эта строка будет переведена в нижний регистр, если только она не заключена в кавычки. Например:
nextval('foo') обращается к последовательностиfoo
nextval('FOO') обращается к последовательностиfoo
nextval('"Foo"') обращается к последовательностиFoo
При необходимости имя последовательности можно дополнить именем схемы:
nextval('myschema.foo') обращается кmyschema.foo
nextval('"myschema".foo') то же самое nextval('foo') ищетfoo
в пути поиска
Подробнее тип regclass
описан в Разделе 8.18.
Примечание
В PostgreSQL до версии 8.1 аргументы этих функций имели тип text
, а не regclass
, и поэтому описанное выше преобразование текстовой строки в OID имело место при каждом вызове функции. Это поведение сохраняется и сейчас для обратной совместимости, но сейчас оно реализовано как неявное приведение типа text
к типу regclass
перед вызовом функции.
Когда вы записываете аргумент функции, работающей с последовательностью, как текстовую строку в чистом виде, она становится константой типа regclass
. Так как фактически это будет просто значение OID, оно будет привязано к изначально идентифицированной последовательности, несмотря на то, что она может быть переименована, перенесена в другую схему и т. д. Такое «раннее связывание» обычно желательно для ссылок на последовательности в значениях столбцов по умолчанию и представлениях. Но иногда возникает необходимость в «позднем связывании», когда ссылки на последовательности распознаются в процессе выполнения. Чтобы получить такое поведение, нужно принудительно изменить тип константы с regclass
на text
:
nextval('foo'::text) foo
распознаётся во время выполнения
Заметьте, что версии PostgreSQL до 8.1 поддерживали только позднее связывание, так что это может быть полезно и для совместимости со старыми приложениями.
Конечно же, аргументом таких функций может быть не только константа, но и выражение. Если это выражение текстового типа, неявное приведение типов повлечёт разрешение имени во время выполнения.
Ниже описаны все функции, предназначенные для работы с последовательностями:
nextval
Продвигает последовательность к следующему значению и возвращает его. Это атомарная операция: если
nextval
вызывается одновременно в нескольких сеансах, в результате каждого вызова будут гарантированно получены разные значения.Если последовательность создаётся с параметрами по умолчанию, успешные вызовы
nextval
получают очередные значения по возрастанию, начиная с 1. Другое поведение можно получить с помощью специальных параметров в команде CREATE SEQUENCE; подробнее это описано на странице описания команды.Важно
Во избежание блокирования параллельных транзакций, пытающихся получить значения одной последовательности, операция
nextval
никогда не откатывается; то есть, как только значение было выбрано, оно считается использованным и не будет возвращено снова. Это утверждение верно, даже когда окружающая транзакция впоследствии прерывается или вызывающий запрос никак не использует это значение. Например, командаINSERT
с предложениемON CONFLICT
вычислит кортеж, претендующий на добавление, произведя все требуемые вызовыnextval
, прежде чем выявит конфликты, которые могут привести к отработке правилON CONFLICT
вместо добавления. В таких ситуациях в последовательности задействованных значений могут образовываться «дыры». Таким образом, объекты последовательностей PostgreSQL не годятся для получения непрерывных последовательностей.currval
Возвращает значение, выданное при последнем вызове
nextval
для этой последовательности в текущем сеансе. (Если в данном сеансеnextval
ни разу не вызывалась для данной последовательности, возвращается ошибка.) Так как это значение ограничено рамками сеанса, эта функция выдаёт предсказуемый результат вне зависимости от того, вызвалась ли впоследствииnextval
в других сеансах или нет.lastval
Возвращает значение, выданное при последнем вызове
nextval
в текущем сеансе. Эта функция подобнаcurrval
, но она не принимает в параметрах имя последовательности, а обращается к той последовательности, для которой вызываласьnextval
в последний раз в текущем сеансе. Если в текущем сеансе функцияnextval
ещё не вызывалась, при вызовеlastval
произойдёт ошибка.setval
Сбрасывает счётчик последовательности. В форме с двумя параметрами устанавливает для последовательности заданное значение поля
last_value
и значениеtrue
для флагаis_called
, показывающего, что при следующем вызовеnextval
последовательность должна сначала продвинуться к очередному значению, которое будет возвращено. При этомcurrval
также возвратит заданное значение. В форме с тремя параметрами флагуis_called
можно присвоитьtrue
илиfalse
. Со значениемtrue
она действует так же, как и форма с двумя параметрами. Если же присвоить этому флагу значениеfalse
, первый вызовnextval
после этого вернёт именно заданное значение, а продвижение последовательности произойдёт при последующем вызовеnextval
. Кроме того, значение, возвращаемоеcurrval
в этом случае, не меняется. Например,SELECT setval('foo', 42); Следующий вызов
nextval
вернёт 43 SELECT setval('foo', 42, true); То же самое SELECT setval('foo', 42, false); Следующий вызовnextval
вернёт 42Результатом самой функции
setval
будет просто значение её второго аргумента.Важно
Так как значения последовательностей изменяются вне транзакций, действие функции
setval
не отменяется при откате транзакции.
32.21. Building libpq Programs #
To build (i.e., compile and link) a program using libpq you need to do all of the following things:
Include the
libpq-fe.h
header file:#include <libpq-fe.h>
If you failed to do that then you will normally get error messages from your compiler similar to:
foo.c: In function `main': foo.c:34: `PGconn' undeclared (first use in this function) foo.c:35: `PGresult' undeclared (first use in this function) foo.c:54: `CONNECTION_BAD' undeclared (first use in this function) foo.c:68: `PGRES_COMMAND_OK' undeclared (first use in this function) foo.c:95: `PGRES_TUPLES_OK' undeclared (first use in this function)
Point your compiler to the directory where the Postgres Pro header files were installed, by supplying the
-I
option to your compiler. (In some cases the compiler will look into the directory in question by default, so you can omit this option.) For instance, your compile command line could look like:directory
cc -c -I/usr/local/pgsql/include testprog.c
If you are using makefiles then add the option to the
CPPFLAGS
variable:CPPFLAGS += -I/usr/local/pgsql/include
If there is any chance that your program might be compiled by other users then you should not hardcode the directory location like that. Instead, you can run the utility
pg_config
to find out where the header files are on the local system:$
pg_config --includedir/usr/local/include
If you have
pkg-config
installed, you can run instead:$
pkg-config --cflags libpq-I/usr/local/include
Note that this will already include the
-I
in front of the path.Failure to specify the correct option to the compiler will result in an error message such as:
testlibpq.c:8:22: libpq-fe.h: No such file or directory
When linking the final program, specify the option
-lpq
so that the libpq library gets pulled in, as well as the option-L
to point the compiler to the directory where the libpq library resides. (Again, the compiler will search some directories by default.) For maximum portability, put thedirectory
-L
option before the-lpq
option. For example:cc -o testprog testprog1.o testprog2.o -L/usr/local/pgsql/lib -lpq
You can find out the library directory using
pg_config
as well:$
pg_config --libdir/usr/local/pgsql/lib
Or again use
pkg-config
:$
pkg-config --libs libpq-L/usr/local/pgsql/lib -lpq
Note again that this prints the full options, not only the path.
Error messages that point to problems in this area could look like the following:
testlibpq.o: In function `main': testlibpq.o(.text+0x60): undefined reference to `PQsetdbLogin' testlibpq.o(.text+0x71): undefined reference to `PQstatus' testlibpq.o(.text+0xa4): undefined reference to `PQerrorMessage'
This means you forgot
-lpq
./usr/bin/ld: cannot find -lpq
This means you forgot the
-L
option or did not specify the right directory.