40.1. Обзор
PL/pgSQL это процедурный язык для СУБД Postgres Pro. Целью проектирования PL/pgSQL было создание загружаемого процедурного языка, который:
используется для создания функций и триггеров,
добавляет управляющие структуры к языку SQL,
может выполнять сложные вычисления,
наследует все пользовательские типы, функции и операторы,
может быть определён как доверенный язык,
прост в использовании.
Функции PL/pgSQL могут использоваться везде, где допустимы встроенные функции. Например, можно создать функции со сложными вычислениями и условной логикой, а затем использовать их при определении операторов или в индексных выражениях.
В версии PostgreSQL 9.0 и выше PL/pgSQL устанавливается по умолчанию. Тем не менее это по-прежнему загружаемый модуль и администраторы, особо заботящиеся о безопасности, могут удалить его при необходимости.
40.1.1. Преимущества использования PL/pgSQL
Postgres Pro и большинство других СУБД используют SQL в качестве языка запросов. SQL хорошо переносим и прост в изучении. Однако каждый оператор SQL выполняется индивидуально на сервере базы данных.
Это значит, что ваше клиентское приложение должно каждый запрос отправлять на сервер, ждать пока он будет обработан, получать результат, делать некоторые вычисления, затем отправлять последующие запросы на сервер. Всё это требует межпроцессного взаимодействия, а также несёт нагрузку на сеть, если клиент и сервер базы данных расположены на разных компьютерах.
PL/pgSQL позволяет сгруппировать блок вычислений и последовательность запросов внутри сервера базы данных, таким образом, мы получаем силу процедурного языка и простоту использования SQL при значительной экономии накладных расходов на клиент-серверное взаимодействие.
Исключаются дополнительные обращения между клиентом и сервером
Промежуточные ненужные результаты не передаются между сервером и клиентом
Есть возможность избежать многочисленных разборов одного запроса
В результате это приводит к значительному увеличению производительности по сравнению с приложением, которое не использует хранимых функций.
Кроме того, PL/pgSQL позволяет использовать все типы данных, операторы и функции SQL.
40.1.2. Поддерживаемые типы данных аргументов и возвращаемых значений
Функции на PL/pgSQL могут принимать в качестве аргументов все поддерживаемые сервером скалярные типы данных или массивы и возвращать в качестве результата любой из этих типов. Они могут принимать и возвращать любой именованный составной тип (тип кортежа). Также есть возможность объявить функцию на PL/pgSQL как возвращающую record
, то есть её результатом будет тип кортежа, столбцы которого определит спецификация вызывающего запроса, как описано в Подразделе 7.2.1.4.
Использование маркера VARIADIC
позволяет объявлять функции на PL/pgSQL с переменным числом аргументов. Это работает точно так же, как и для функций на SQL, как описано в Подразделе 35.4.5.
Функции на PL/pgSQL могут принимать и возвращать полиморфные типы anyelement
, anyarray
, anynonarray
, anyenum
и anyrange
. В таких случаях фактические типы данных могут меняться от вызова к вызову, как описано в Подраздел 35.2.5. Пример показан в Подразделе 40.3.1.
Функции на PL/pgSQL могут возвращать «множества» (или таблицы) любого типа, которые могут быть возвращены в виде одного объекта. Такие функции генерируют вывод, выполняя команду RETURN NEXT
для каждого элемента результирующего набора или RETURN QUERY
для вывода результата запроса.
Наконец, при отсутствии полезного возвращаемого значения функция на PL/pgSQL может возвращать void
.
Функции на PL/pgSQL можно объявить с выходными параметрами вместо явного задания типа возвращаемого значения. Это не добавляет никаких фундаментальных возможностей языку, но часто бывает удобно, особенно для возвращения нескольких значений. Нотация RETURNS TABLE
может использоваться вместо RETURNS SETOF
.
Конкретные примеры рассматриваются в Подразделе 40.3.1 и Подразделе 40.6.1.
45.4. Expressions
All expressions used in PL/pgSQL statements are processed using the server's main SQL executor. For example, when you write a PL/pgSQL statement like
IF expression
THEN ...
PL/pgSQL will evaluate the expression by feeding a query like
SELECT expression
to the main SQL engine. While forming the SELECT
command, any occurrences of PL/pgSQL variable names are replaced by query parameters, as discussed in detail in Section 45.12.1. This allows the query plan for the SELECT
to be prepared just once and then reused for subsequent evaluations with different values of the variables. Thus, what really happens on first use of an expression is essentially a PREPARE
command. For example, if we have declared two integer variables x
and y
, and we write
IF x < y THEN ...
what happens behind the scenes is equivalent to
PREPARE statement_name
(integer, integer) AS SELECT $1 < $2;
and then this prepared statement is EXECUTE
d for each execution of the IF
statement, with the current values of the PL/pgSQL variables supplied as parameter values. Normally these details are not important to a PL/pgSQL user, but they are useful to know when trying to diagnose a problem. More information appears in Section 45.12.2.
Since an expression
is converted to a SELECT
command, it can contain the same clauses that an ordinary SELECT
would, except that it cannot include a top-level UNION
, INTERSECT
, or EXCEPT
clause. Thus for example one could test whether a table is non-empty with
IF count(*) > 0 FROM my_table THEN ...
since the expression
between IF
and THEN
is parsed as though it were SELECT count(*) > 0 FROM my_table
. The SELECT
must produce a single column, and not more than one row. (If it produces no rows, the result is taken as NULL.)