xfunc.sgml
上传用户:blenddy
上传日期:2007-01-07
资源大小:6495k
文件大小:20k
源码类别:

数据库系统

开发平台:

Unix_Linux

  1. <Chapter Id="xfunc">
  2. <Title>Extending <Acronym>SQL</Acronym>: Functions</Title>
  3. <Para>
  4.      As  it  turns  out,  part of defining a new type is the
  5.      definition of functions  that  describe  its  behavior.
  6.      Consequently,  while  it  is  possible  to define a new
  7.      function without defining a new type,  the  reverse  is
  8.      not  true.   We therefore describe how to add new functions 
  9.      to <ProductName>Postgres</ProductName> before  describing  
  10.      how  to  add  new types.
  11.      <ProductName>Postgres</ProductName>  <Acronym>SQL</Acronym>  
  12.      provides  two  types of functions: query language functions 
  13.      (functions written in <Acronym>SQL</Acronym>  and  programming  
  14.      language  functions  (functions  written in a compiled 
  15.      programming language such as <Acronym>C</Acronym>.)  Either  kind
  16.      of  function  can take a base type, a composite type or
  17.      some combination as arguments (parameters).   In  addition, 
  18.      both kinds of functions can return a base type or
  19.      a composite type.  It's easier to define <Acronym>SQL</Acronym> 
  20.      functions, so we'll start with those.  Examples in this section 
  21.      can also be found in <FileName>funcs.sql</FileName> 
  22.      and <FileName>funcs.c</FileName>.
  23. </Para>
  24. <Sect1>
  25. <Title>Query Language (<Acronym>SQL</Acronym>) Functions</Title>
  26. <Sect2>
  27. <Title><Acronym>SQL</Acronym> Functions on Base Types</Title>
  28. <Para>
  29.      The simplest possible <Acronym>SQL</Acronym> function has no arguments and
  30.      simply returns a base type, such as <Acronym>int4</Acronym>:
  31.      
  32. <ProgramListing>
  33.     CREATE FUNCTION one() RETURNS int4
  34.      AS 'SELECT 1 as RESULT' LANGUAGE 'sql';
  35.     SELECT one() AS answer;
  36.          +-------+
  37.          |answer |
  38.          +-------+
  39.          |1      |
  40.          +-------+
  41. </ProgramListing>
  42. </Para>
  43. <Para>
  44.      Notice that we defined a target list for  the  function
  45.      (with  the  name  RESULT),  but  the target list of the
  46.      query that invoked the function overrode the function's
  47.      target  list.   Hence,  the  result  is labelled answer
  48.      instead of one.
  49. </Para>
  50. <Para>
  51.      It's almost as easy to define <Acronym>SQL</Acronym> functions  
  52.      that take base types as arguments.  In the example below, notice
  53.      how we refer to the arguments within the function as $1
  54.      and $2.
  55.      
  56. <ProgramListing>
  57.     CREATE FUNCTION add_em(int4, int4) RETURNS int4
  58.      AS 'SELECT $1 + $2;' LANGUAGE 'sql';
  59.     SELECT add_em(1, 2) AS answer;
  60.          +-------+
  61.          |answer |
  62.          +-------+
  63.          |3      |
  64.          +-------+
  65. </ProgramListing>
  66. </Para>
  67. </sect2>
  68. <Sect2>
  69. <Title><Acronym>SQL</Acronym> Functions on Composite Types</Title>
  70. <Para>
  71.      When  specifying  functions with arguments of composite
  72.      types (such as EMP), we must  not  only  specify  which
  73.      argument  we  want (as we did above with $1 and $2) but
  74.      also the attributes of  that  argument.   For  example,
  75.      take the function double_salary that computes what your
  76.      salary would be if it were doubled.
  77.      
  78. <ProgramListing>
  79.     CREATE FUNCTION double_salary(EMP) RETURNS int4
  80.      AS 'SELECT $1.salary * 2 AS salary;' LANGUAGE 'sql';
  81.     SELECT name, double_salary(EMP) AS dream
  82.      FROM EMP
  83.      WHERE EMP.cubicle ~= '(2,1)'::point;
  84.      
  85.          +-----+-------+
  86.          |name | dream |
  87.          +-----+-------+
  88.          |Sam  | 2400  |
  89.          +-----+-------+
  90. </ProgramListing>
  91. </para>
  92. <Para>
  93.      Notice the use of the syntax $1.salary.
  94.      Before launching into the  subject  of  functions  that
  95.      return  composite  types,  we  must first introduce the
  96.      function notation for projecting attributes.  The  simple  way 
  97.      to explain this is that we can usually use the
  98.      notation attribute(class)  and  class.attribute  interchangably.
  99.      
  100. <ProgramListing>
  101.     --
  102.     -- this is the same as:
  103.     --  SELECT EMP.name AS youngster FROM EMP WHERE EMP.age &lt; 30
  104.     --
  105.     SELECT name(EMP) AS youngster
  106.      FROM EMP
  107.      WHERE age(EMP) &lt; 30;
  108.          +----------+
  109.          |youngster |
  110.          +----------+
  111.          |Sam       |
  112.          +----------+
  113. </ProgramListing>
  114. </para>
  115. <Para>
  116.      As  we shall see, however, this is not always the case.
  117.      This function notation is important when we want to use
  118.      a  function that returns a single instance.  We do this
  119.      by assembling the entire instance within the  function,
  120.      attribute  by attribute.  This is an example of a function 
  121.      that returns a single EMP instance:
  122.      
  123. <ProgramListing>
  124.     CREATE FUNCTION new_emp() RETURNS EMP
  125.      AS 'SELECT 'None'::text AS name,
  126.       1000 AS salary,
  127.       25 AS age,
  128.        '(2,2)'::point AS cubicle'
  129.       LANGUAGE 'sql';
  130. </ProgramListing>
  131. </Para>
  132. <Para>
  133.      In this case we have specified each of  the  attributes
  134.      with  a  constant value, but any computation or expression 
  135.      could have been substituted for these constants.
  136.      Defining a function like this can be tricky.   Some  of
  137.      the more important caveats are as follows:
  138.      
  139.      
  140. <ItemizedList>
  141. <ListItem>
  142. <Para>
  143.      The  target  list  order must be exactly the same as
  144.      that in which the attributes appear  in  the  CREATE
  145.      TABLE statement (or when you execute a .*  query).
  146. </Para>
  147.       </ListItem>
  148.       <ListItem>
  149. <Para>
  150. You must typecast the expressions (using ::) very carefully 
  151. or you will see  the  following error:
  152.         
  153. <ProgramListing>
  154.    WARN::function declared to return type EMP does not retrieve (EMP.*)
  155. </ProgramListing>
  156. </Para>
  157.       </ListItem>
  158. <ListItem>
  159. <Para>
  160. When calling a function that returns an instance, we
  161.         cannot retrieve the entire instance.  We must either
  162.         project an attribute out of the instance or pass the
  163.         entire instance into another function.
  164. <ProgramListing>
  165.     SELECT name(new_emp()) AS nobody;
  166.             +-------+
  167.             |nobody |
  168.             +-------+
  169.             |None   |
  170.             +-------+
  171. </ProgramListing>
  172. </Para>
  173.       </ListItem>
  174. <ListItem>
  175. <Para>
  176. The reason why, in general, we must use the function
  177.         syntax  for projecting attributes of function return
  178.         values is that the parser  just  doesn't  understand
  179.         the  other (dot) syntax for projection when combined
  180.         with function calls.
  181.         
  182. <ProgramListing>
  183.             SELECT new_emp().name AS nobody;
  184.             WARN:parser: syntax error at or near "."
  185. </ProgramListing>
  186. </Para>
  187.       </ListItem>
  188. </ItemizedList>
  189. </para>     
  190. <Para>
  191.      Any collection of commands in the  <Acronym>SQL</Acronym>  query  
  192.      language can be packaged together and defined as a function.
  193.      The commands can include updates (i.e., <Acronym>insert</Acronym>,  
  194.      <Acronym>update</Acronym> and <Acronym>delete</Acronym>) as well 
  195.      as <Acronym>select</Acronym> queries.  However, the final command 
  196.      must be a <Acronym>select</Acronym> that returns whatever is
  197.      specified as the function's returntype.
  198.      
  199. <ProgramListing>
  200.     CREATE FUNCTION clean_EMP () RETURNS int4
  201.      AS 'DELETE FROM EMP WHERE EMP.salary &lt;= 0;
  202.     SELECT 1 AS ignore_this'
  203.      LANGUAGE 'sql';
  204.     SELECT clean_EMP();
  205.          +--+
  206.          |x |
  207.          +--+
  208.          |1 |
  209.          +--+
  210.          
  211. </ProgramListing>
  212. </Para>
  213. </sect2>
  214. </sect1>
  215. <Sect1>
  216. <Title>Programming Language Functions</Title>
  217. <Sect2>
  218. <Title>Programming Language Functions on Base Types</Title>
  219. <Para>
  220.      Internally, <ProductName>Postgres</ProductName> regards a
  221.      base type as a "blob  of memory."   The  user-defined  
  222.      functions that you define over a type in turn define the 
  223.      way  that  <ProductName>Postgres</ProductName> can operate  
  224.      on  it.  That is, <ProductName>Postgres</ProductName> will 
  225.      only store and retrieve the data from disk and use  your  
  226.      user-defined functions to input, process, and output the data.
  227.      Base types can have one of three internal formats:
  228. <ItemizedList>
  229. <ListItem><Para>pass by value, fixed-length</Para>
  230.       </ListItem>
  231. <ListItem><Para>pass by reference, fixed-length</Para>
  232.       </ListItem>
  233. <ListItem><Para>pass by reference, variable-length</Para>
  234.       </ListItem>
  235. </ItemizedList>
  236. </Para>
  237. <Para>
  238.      By-value  types  can  only be 1, 2 or 4 bytes in length
  239.      (even if your computer supports by-value types of other
  240.      sizes).   <ProductName>Postgres</ProductName>  itself 
  241.      only passes integer types by value.  You should be careful 
  242.      to define your types such that  they  will  be  the  same  
  243.      size (in bytes) on all architectures.  For example, the 
  244.      <Acronym>long</Acronym> type is dangerous because  it  
  245.      is 4 bytes on some machines and 8 bytes on others, whereas 
  246.      <Acronym>int</Acronym>  type  is  4  bytes  on  most  
  247.      <Acronym>UNIX</Acronym> machines  (though  not  on most 
  248.      personal computers).  A reasonable implementation of  
  249.      the  <Acronym>int4</Acronym>  type  on  <Acronym>UNIX</Acronym>
  250.      machines might be:
  251.      
  252. <ProgramListing>
  253.     /* 4-byte integer, passed by value */
  254.     typedef int int4;
  255. </ProgramListing>
  256. </Para>
  257. <Para>
  258.      On  the  other hand, fixed-length types of any size may
  259.      be passed by-reference.  For example, here is a  sample
  260.      implementation of a <ProductName>Postgres</ProductName> type:
  261.      
  262. <ProgramListing>
  263.          /* 16-byte structure, passed by reference */
  264.     typedef struct
  265.     {
  266.         double  x, y;
  267.     } Point;
  268. </ProgramListing>
  269. </Para>
  270. <Para>
  271.      Only  pointers  to  such types can be used when passing
  272.      them in and out of <ProductName>Postgres</ProductName> functions.
  273.      Finally, all variable-length types must also be  passed
  274.      by  reference.   All  variable-length  types must begin
  275.      with a length field of exactly 4 bytes, and all data to
  276.      be  stored within that type must be located in the memory 
  277.      immediately  following  that  length  field.   The
  278.      length  field  is  the  total  length  of the structure
  279.      (i.e.,  it  includes  the  size  of  the  length  field
  280.      itself).  We can define the text type as follows:
  281. </Para>
  282. <Para>
  283. <ProgramListing>
  284.          typedef struct {
  285.              int4 length;
  286.              char data[1];
  287.          } text;
  288. </ProgramListing>
  289. </Para>
  290. <Para>
  291.      Obviously,  the  data  field is not long enough to hold
  292.      all possible strings -- it's impossible to declare such
  293.      a  structure  in  <Acronym>C</Acronym>.  When manipulating 
  294.      variable-length types, we must  be  careful  to  allocate  
  295.      the  correct amount  of memory and initialize the length field.  
  296.      For example, if we wanted to  store  40  bytes  in  a  text
  297.      structure, we might use a code fragment like this:
  298. <ProgramListing>
  299.          #include "postgres.h"
  300.          ...
  301.          char buffer[40]; /* our source data */
  302.          ...
  303.          text *destination = (text *) palloc(VARHDRSZ + 40);
  304.          destination-&gt;length = VARHDRSZ + 40;
  305.          memmove(destination-&gt;data, buffer, 40);
  306.          ...
  307. </ProgramListing>
  308. </Para>
  309. <Para>
  310.      Now that we've gone over all of the possible structures
  311.      for base types, we can show some examples of real functions. 
  312.      Suppose <FileName>funcs.c</FileName> look like:
  313. <ProgramListing>
  314.          #include &lt;string.h&gt;
  315.          #include "postgres.h"
  316.          /* By Value */
  317.          
  318.          int
  319.          add_one(int arg)
  320.          {
  321.              return(arg + 1);
  322.          }
  323.          
  324.          /* By Reference, Fixed Length */
  325.          
  326.          Point *
  327.          makepoint(Point *pointx, Point *pointy )
  328.          {
  329.              Point     *new_point = (Point *) palloc(sizeof(Point));
  330.         
  331.              new_point->x = pointx->x;
  332.              new_point->y = pointy->y;
  333.                 
  334.              return new_point;
  335.          }
  336.         
  337.          /* By Reference, Variable Length */
  338.          
  339.          text *
  340.          copytext(text *t)
  341.          {
  342.              /*
  343.               * VARSIZE is the total size of the struct in bytes.
  344.               */
  345.              text *new_t = (text *) palloc(VARSIZE(t));
  346.              memset(new_t, 0, VARSIZE(t));
  347.              VARSIZE(new_t) = VARSIZE(t);
  348.              /*
  349.               * VARDATA is a pointer to the data region of the struct.
  350.               */
  351.              memcpy((void *) VARDATA(new_t), /* destination */
  352.                     (void *) VARDATA(t),     /* source */
  353.                     VARSIZE(t)-VARHDRSZ);        /* how many bytes */
  354.              return(new_t);
  355.          }
  356.          
  357.          text *
  358.          concat_text(text *arg1, text *arg2)
  359.          {
  360.              int32 new_text_size = VARSIZE(arg1) + VARSIZE(arg2) - VARHDRSZ;
  361.              text *new_text = (text *) palloc(new_text_size);
  362.              memset((void *) new_text, 0, new_text_size);
  363.              VARSIZE(new_text) = new_text_size;
  364.              strncpy(VARDATA(new_text), VARDATA(arg1), VARSIZE(arg1)-VARHDRSZ);
  365.              strncat(VARDATA(new_text), VARDATA(arg2), VARSIZE(arg2)-VARHDRSZ);
  366.              return (new_text);
  367.          }
  368. </ProgramListing>
  369. </Para>
  370. <Para>
  371.      On <Acronym>OSF/1</Acronym> we would type:
  372.      
  373. <ProgramListing>
  374.          CREATE FUNCTION add_one(int4) RETURNS int4
  375.               AS 'PGROOT/tutorial/funcs.so' LANGUAGE 'c';
  376.          CREATE FUNCTION makepoint(point, point) RETURNS point
  377.               AS 'PGROOT/tutorial/funcs.so' LANGUAGE 'c';
  378.     
  379.          CREATE FUNCTION concat_text(text, text) RETURNS text
  380.               AS 'PGROOT/tutorial/funcs.so' LANGUAGE 'c';
  381.                                   
  382.          CREATE FUNCTION copytext(text) RETURNS text
  383.               AS 'PGROOT/tutorial/funcs.so' LANGUAGE 'c';
  384. </ProgramListing>
  385. </Para>
  386. <Para>
  387.      On  other  systems,  we might have to make the filename
  388.      end in .sl (to indicate that it's a shared library).
  389. </Para>
  390. </Sect2>
  391. <Sect2>
  392. <Title>Programming Language Functions on Composite Types</Title>
  393. <Para>
  394.      Composite types do not  have  a  fixed  layout  like  C
  395.      structures.   Instances of a composite type may contain
  396.      null fields.  In addition,  composite  types  that  are
  397.      part  of  an  inheritance  hierarchy may have different
  398.      fields than other members of the same inheritance hierarchy.    
  399.      Therefore,  <ProductName>Postgres</ProductName>  provides  
  400.      a  procedural interface for accessing fields of composite types  
  401.      from C.  As <ProductName>Postgres</ProductName> processes 
  402.      a set of instances, each instance will be passed into your 
  403.      function as an  opaque  structure of type <Acronym>TUPLE</Acronym>.
  404.      Suppose we want to write a function to answer the query
  405. <ProgramListing>
  406.          * SELECT name, c_overpaid(EMP, 1500) AS overpaid
  407.            FROM EMP
  408.            WHERE name = 'Bill' or name = 'Sam';
  409. </ProgramListing>
  410.      In the query above, we can define c_overpaid as:
  411.      
  412. <ProgramListing>
  413.          #include "postgres.h"
  414.          #include "executor/executor.h"  /* for GetAttributeByName() */
  415.          
  416.          bool
  417.          c_overpaid(TupleTableSlot *t, /* the current instance of EMP */
  418.                     int4 limit)
  419.          {
  420.              bool isnull = false;
  421.              int4 salary;
  422.              salary = (int4) GetAttributeByName(t, "salary", &amp;isnull);
  423.              if (isnull)
  424.                  return (false);
  425.              return(salary &gt; limit);
  426.          }
  427. </ProgramListing>
  428. </Para>
  429. <Para>
  430.      <Acronym>GetAttributeByName</Acronym> is the 
  431.      <ProductName>Postgres</ProductName> system function that
  432.      returns attributes out of the current instance.  It has
  433.      three arguments: the argument of type TUPLE passed into
  434.      the  function, the name of the desired attribute, and a
  435.      return parameter that describes whether  the  attribute
  436.      is  null.   <Acronym>GetAttributeByName</Acronym> will 
  437.      align data properly so you can cast its return value to 
  438.      the  desired  type. For  example, if you have an attribute 
  439.      name which is of the type name, the <Acronym>GetAttributeByName</Acronym> 
  440.      call would look like:
  441. <ProgramListing>
  442.          char *str;
  443.          ...
  444.          str = (char *) GetAttributeByName(t, "name", &amp;isnull)
  445. </ProgramListing>
  446. </Para>
  447. <Para>
  448.      The  following  query  lets  <ProductName>Postgres</ProductName>  
  449.      know  about  the c_overpaid function:
  450. <ProgramListing>
  451.          * CREATE FUNCTION c_overpaid(EMP, int4) RETURNS bool
  452.               AS 'PGROOT/tutorial/obj/funcs.so' LANGUAGE 'c';
  453. </ProgramListing>
  454. </Para>
  455. <Para>
  456.      While there are ways to construct new instances or modify  
  457.      existing instances from within a C function, these
  458.      are far too complex to discuss in this manual.
  459. </Para>
  460. </Sect2>
  461. <Sect2>
  462. <Title>Caveats</Title>
  463. <Para>
  464.      We now turn to the more difficult task of writing  
  465.      programming  language  functions.  Be warned: this section
  466.      of the manual will not make you a programmer.  You must
  467.      have  a  good  understanding of <Acronym>C</Acronym> 
  468.      (including the use of pointers and the malloc memory manager)  
  469.      before  trying to write <Acronym>C</Acronym> functions for 
  470.      use with <ProductName>Postgres</ProductName>. While  it may 
  471.      be possible to load functions written in languages other 
  472.      than <Acronym>C</Acronym> into  <ProductName>Postgres</ProductName>,  
  473.      this  is  often difficult  (when  it  is possible at all) 
  474.      because other languages, such as <Acronym>FORTRAN</Acronym> 
  475.      and <Acronym>Pascal</Acronym> often do not follow the same 
  476.      "calling convention" as <Acronym>C</Acronym>.  That is, other
  477.      languages  do  not  pass  argument  and  return  values
  478.      between functions in the same way.  For this reason, we
  479.      will assume that your  programming  language  functions
  480.      are written in <Acronym>C</Acronym>.
  481.      The  basic  rules  for building <Acronym>C</Acronym> functions 
  482.      are as follows:
  483. <ItemizedList>
  484. <ListItem>
  485. <Para>
  486.             Most of the header (include) files for 
  487.             <ProductName>Postgres</ProductName>
  488.             should      already      be     installed     in
  489.             <FileName>PGROOT/include</FileName>  (see  Figure  2).
  490.             You should always include
  491.             
  492. <ProgramListing>
  493.                 -I$PGROOT/include
  494. </ProgramListing>
  495.             on  your  cc  command lines.  Sometimes, you may
  496.             find that you require header files that  are  in
  497.             the  server source itself (i.e., you need a file
  498.             we neglected to install in include).   In  those
  499.             cases you may need to add one or more of
  500. <ProgramListing>
  501.                 -I$PGROOT/src/backend
  502.                 -I$PGROOT/src/backend/include
  503.                 -I$PGROOT/src/backend/port/&lt;PORTNAME&gt;
  504.                 -I$PGROOT/src/backend/obj
  505. </ProgramListing>
  506.             (where &lt;PORTNAME&gt; is the name of the port, e.g.,
  507.             alpha or sparc).
  508. </para>
  509. </ListItem>
  510. <ListItem>
  511. <Para>      When allocating memory, use  the
  512.             <ProductName>Postgres</ProductName>
  513.             routines  palloc  and  pfree  instead of the 
  514.             corresponding <Acronym>C</Acronym> library  routines  
  515.             malloc  and  free.
  516.             The  memory  allocated  by  palloc will be freed
  517.             automatically at the end  of  each  transaction,
  518.             preventing memory leaks.
  519. </Para>
  520.       </ListItem>
  521.       <ListItem>
  522. <Para>   Always  zero  the bytes of your structures using
  523.             memset or bzero.  Several routines (such as  the
  524.             hash access method, hash join and the sort algorithm) 
  525.             compute functions of the  raw  bits  contained  in 
  526.             your structure.  Even if you initialize all fields 
  527.             of your structure, there  may  be
  528.             several bytes of alignment padding (holes in the
  529.             structure) that may contain garbage values.
  530. </Para>
  531.       </ListItem>
  532.       <ListItem>
  533. <Para>      Most of the internal <ProductName>Postgres</ProductName> 
  534.             types are declared in  postgres.h,  so  it's a good 
  535.             idea to always include that file as well.  Including 
  536.             postgres.h will also include elog.h and palloc.h for you.
  537. </Para>
  538.       </ListItem>
  539.       <ListItem>
  540. <Para>      Compiling and loading your object code  so  that
  541.             it  can  be  dynamically  loaded  into  
  542.             <ProductName>Postgres</ProductName>
  543.             always requires special flags.  See  Appendix  A
  544.             for  a  detailed explanation of how to do it for
  545.             your particular operating system.
  546. </Para>
  547. </ListItem>
  548. </ItemizedList>
  549. </Para>
  550. </Sect2>
  551. </sect1>
  552. </chapter>