mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
149 lines
5.5 KiB
HTML
149 lines
5.5 KiB
HTML
<HTML>
|
|
<HEAD>
|
|
<TITLE>The POSTGRES95 User Manual - EXTENDING SQL: TYPES</TITLE>
|
|
</HEAD>
|
|
|
|
<BODY>
|
|
|
|
<font size=-1>
|
|
<A HREF="pg95user.html">[ TOC ]</A>
|
|
<A HREF="xfunc.html">[ Previous ]</A>
|
|
<A HREF="xoper.html">[ Next ]</A>
|
|
</font>
|
|
<HR>
|
|
<H1>8. EXTENDING SQL: TYPES</H1>
|
|
<HR>
|
|
As previously mentioned, there are two kinds of types
|
|
in POSTGRES: base types (defined in a programming language)
|
|
and composite types (instances).
|
|
Examples in this section up to interfacing indices can
|
|
be found in <CODE>complex.sql</CODE> and <CODE>complex.c</CODE>. Composite examples
|
|
are in <CODE>funcs.sql</CODE>.
|
|
<p>
|
|
<H2><A NAME="user-defined-types">8.1. User-Defined Types</A></H2>
|
|
<p>
|
|
<H3><A NAME="functions-needed-for-a-user-defined-type">8.1.1. Functions Needed for a User-Defined Type</A></H3>
|
|
A user-defined type must always have input and output
|
|
functions. These functions determine how the type
|
|
appears in strings (for input by the user and output to
|
|
the user) and how the type is organized in memory. The
|
|
input function takes a null-delimited character string
|
|
as its input and returns the internal (in memory)
|
|
representation of the type. The output function takes the
|
|
internal representation of the type and returns a null
|
|
delimited character string.
|
|
Suppose we want to define a complex type which represents
|
|
complex numbers. Naturally, we choose to represent a
|
|
complex in memory as the following <B>C</B> structure:
|
|
|
|
<pre> typedef struct Complex {
|
|
double x;
|
|
double y;
|
|
} Complex;
|
|
</pre>
|
|
and a string of the form (x,y) as the external string
|
|
representation.
|
|
These functions are usually not hard to write, especially
|
|
the output function. However, there are a number of points
|
|
to remember.
|
|
|
|
<OL>
|
|
<LI> When defining your external (string) representation,
|
|
remember that you must eventually write a
|
|
complete and robust parser for that representation
|
|
as your input function!
|
|
|
|
<pre> Complex *
|
|
complex_in(char *str)
|
|
{
|
|
double x, y;
|
|
Complex *result;
|
|
|
|
if (sscanf(str, " ( %lf , %lf )", &x, &y) != 2) {
|
|
elog(WARN, "complex_in: error in parsing
|
|
return NULL;
|
|
}
|
|
result = (Complex *)palloc(sizeof(Complex));
|
|
result->x = x;
|
|
result->y = y;
|
|
return (result);
|
|
}
|
|
</pre>
|
|
|
|
The output function can simply be:
|
|
|
|
<pre> char *
|
|
complex_out(Complex *complex)
|
|
{
|
|
char *result;
|
|
<p>
|
|
if (complex == NULL)
|
|
return(NULL);
|
|
<p>
|
|
result = (char *) palloc(60);
|
|
sprintf(result, "(%g,%g)", complex->x, complex->y);
|
|
return(result);
|
|
}
|
|
</pre>
|
|
<LI> You should try to make the input and output
|
|
functions inverses of each other. If you do
|
|
not, you will have severe problems when you need
|
|
to dump your data into a file and then read it
|
|
back in (say, into someone else's database on
|
|
another computer). This is a particularly common
|
|
problem when floating-point numbers are
|
|
involved.
|
|
</OL>
|
|
To define the <B>complex</B> type, we need to create the two
|
|
user-defined functions complex_in and complex_out
|
|
before creating the type:
|
|
|
|
<pre> CREATE FUNCTION complex_in(opaque)
|
|
RETURNS complex
|
|
AS '/usr/local/postgres95/tutorial/obj/complex.so'
|
|
LANGUAGE 'c';
|
|
|
|
CREATE FUNCTION complex_out(opaque)
|
|
RETURNS opaque
|
|
AS '/usr/local/postgres95/tutorial/obj/complex.so'
|
|
LANGUAGE 'c';
|
|
|
|
CREATE TYPE complex (
|
|
internallength = 16,
|
|
input = complex_in,
|
|
output = complex_out
|
|
);
|
|
</pre>
|
|
|
|
As discussed earlier, POSTGRES fully supports arrays of
|
|
base types. Additionally, POSTGRES supports arrays of
|
|
user-defined types as well. When you define a type,
|
|
POSTGRES automatically provides support for arrays of
|
|
that type. For historical reasons, the array type has
|
|
the same name as the user-defined type with the
|
|
underscore character _ prepended.
|
|
Composite types do not need any function defined on
|
|
them, since the system already understands what they
|
|
look like inside.
|
|
<p>
|
|
<H3><A NAME="large-objects">8.1.2. Large Objects</A></H3>
|
|
The types discussed to this point are all "small"
|
|
objects -- that is, they are smaller than 8KB<A HREF="#7"><font size=-1>[7]</font></A> in size.
|
|
If you require a larger type for something like a document
|
|
retrieval system or for storing bitmaps, you will
|
|
need to use the POSTGRES large object interface.
|
|
<p>
|
|
<HR>
|
|
<A NAME="8"><B>[7]</B></A> 8 * 1024 == 8192 bytes. In fact, the type must be considerably smaller than 8192 bytes, since the POSTGRES tuple
|
|
and page overhead must also fit into this 8KB limitation.
|
|
The actual value that fits depends on the machine architecture.
|
|
<HR>
|
|
<font size=-1>
|
|
<A HREF="pg95user.html">[ TOC ]</A>
|
|
<A HREF="xfunc.html">[ Previous ]</A>
|
|
<A HREF="xoper.html">[ Next ]</A>
|
|
</font>
|
|
</BODY>
|
|
</HTML>
|
|
|