diff --git a/doc/src/sgml/func.sgml b/doc/src/sgml/func.sgml index 5caf10711ac52a0ddacc9fe37330f604e1c46d7d..f8f63d89f91e6fa0b4601f2a604e7d8030e1b4bc 100644 --- a/doc/src/sgml/func.sgml +++ b/doc/src/sgml/func.sgml @@ -12548,9 +12548,10 @@ AND <para> This section describes functions that possibly return more than one row. - Currently the only functions in this class are series generating functions, - as detailed in <xref linkend="functions-srf-series"> and - <xref linkend="functions-srf-subscripts">. + The most widely used functions in this class are series generating + functions, as detailed in <xref linkend="functions-srf-series"> and + <xref linkend="functions-srf-subscripts">. Other, more specialized + set-returning functions are described elsewhere in this manual. </para> <table id="functions-srf-series"> diff --git a/doc/src/sgml/xfunc.sgml b/doc/src/sgml/xfunc.sgml index 7b2d2b0ad39944ff0f8f567b889a0f26dd3aeffc..85539feb0d255daf486246dee911802b1a274e17 100644 --- a/doc/src/sgml/xfunc.sgml +++ b/doc/src/sgml/xfunc.sgml @@ -93,8 +93,8 @@ </para> <para> - Alternatively, an SQL function can be declared to return a set, - by specifying the function's return type as <literal>SETOF + Alternatively, an SQL function can be declared to return a set (that is, + multiple rows) by specifying the function's return type as <literal>SETOF <replaceable>sometype</></literal>, or equivalently by declaring it as <literal>RETURNS TABLE(<replaceable>columns</>)</literal>. In this case all rows of the last query's result are returned. Further details appear @@ -927,19 +927,15 @@ SELECT * FROM sum_n_product_with_tab(10); </para> <para> - Currently, functions returning sets can also be called in the select list - of a query. For each row that the query - generates by itself, the function returning set is invoked, and an output - row is generated for each element of the function's result set. Note, - however, that this capability is deprecated and might be removed in future - releases. The following is an example function returning a set from the - select list: + It is frequently useful to construct a query's result by invoking a + set-returning function multiple times, with the parameters for each + invocation coming from successive rows of a table or subquery. The + preferred way to do this is to use the <literal>LATERAL</> key word, + which is described in <xref linkend="queries-lateral">. + Here is an example using a set-returning function to enumerate + elements of a tree structure: <screen> -CREATE FUNCTION listchildren(text) RETURNS SETOF text AS $$ - SELECT name FROM nodes WHERE parent = $1 -$$ LANGUAGE SQL; - SELECT * FROM nodes; name | parent -----------+-------- @@ -951,6 +947,44 @@ SELECT * FROM nodes; SubChild2 | Child1 (6 rows) +CREATE FUNCTION listchildren(text) RETURNS SETOF text AS $$ + SELECT name FROM nodes WHERE parent = $1 +$$ LANGUAGE SQL STABLE; + +SELECT * FROM listchildren('Top'); + listchildren +-------------- + Child1 + Child2 + Child3 +(3 rows) + +SELECT name, child FROM nodes, LATERAL listchildren(name) AS child; + name | child +--------+----------- + Top | Child1 + Top | Child2 + Top | Child3 + Child1 | SubChild1 + Child1 | SubChild2 +(5 rows) +</screen> + + This example does not do anything that we couldn't have done with a + simple join, but in more complex calculations the option to put + some of the work into a function can be quite convenient. + </para> + + <para> + Currently, functions returning sets can also be called in the select list + of a query. For each row that the query + generates by itself, the function returning set is invoked, and an output + row is generated for each element of the function's result set. Note, + however, that this capability is deprecated and might be removed in future + releases. The previous example could also be done with queries like + these: + +<screen> SELECT listchildren('Top'); listchildren -------------- @@ -973,7 +1007,9 @@ SELECT name, listchildren(name) FROM nodes; In the last <command>SELECT</command>, notice that no output row appears for <literal>Child2</>, <literal>Child3</>, etc. This happens because <function>listchildren</function> returns an empty set - for those arguments, so no result rows are generated. + for those arguments, so no result rows are generated. This is the same + behavior as we got from an inner join to the function result when using + the <literal>LATERAL</> syntax. </para> <note> @@ -987,6 +1023,19 @@ SELECT name, listchildren(name) FROM nodes; still happen (and are all completed before returning from the function). </para> </note> + + <note> + <para> + The key problem with using set-returning functions in the select list, + rather than the <literal>FROM</> clause, is that putting more than one + set-returning function in the same select list does not behave very + sensibly. (What you actually get if you do so is a number of output + rows equal to the least common multiple of the numbers of rows produced + by each set-returning function.) The <literal>LATERAL</> syntax + produces less surprising results when calling multiple set-returning + functions, and should usually be used instead. + </para> + </note> </sect2> <sect2 id="xfunc-sql-functions-returning-table">