1 <!-- doc/src/sgml/cube.sgml -->
3 <sect1 id=
"cube" xreflabel=
"cube">
4 <title>cube
— a multi-dimensional cube data type
</title>
6 <indexterm zone=
"cube">
7 <primary>cube (extension)
</primary>
11 This module implements a data type
<type>cube
</type> for
12 representing multidimensional cubes.
16 This module is considered
<quote>trusted
</quote>, that is, it can be
17 installed by non-superusers who have
<literal>CREATE
</literal> privilege
18 on the current database.
21 <sect2 id=
"cube-syntax">
25 <xref linkend=
"cube-repr-table"/> shows the valid external
26 representations for the
<type>cube
</type>
27 type.
<replaceable>x
</replaceable>,
<replaceable>y
</replaceable>, etc. denote
28 floating-point numbers.
31 <table id=
"cube-repr-table">
32 <title>Cube External Representations
</title>
36 <entry>External Syntax
</entry>
37 <entry>Meaning
</entry>
43 <entry><literal><replaceable>x
</replaceable></literal></entry>
44 <entry>A one-dimensional point
45 (or, zero-length one-dimensional interval)
49 <entry><literal>(
<replaceable>x
</replaceable>)
</literal></entry>
50 <entry>Same as above
</entry>
53 <entry><literal><replaceable>x1
</replaceable>,
<replaceable>x2
</replaceable>,...,
<replaceable>xn
</replaceable></literal></entry>
54 <entry>A point in n-dimensional space, represented internally as a
59 <entry><literal>(
<replaceable>x1
</replaceable>,
<replaceable>x2
</replaceable>,...,
<replaceable>xn
</replaceable>)
</literal></entry>
60 <entry>Same as above
</entry>
63 <entry><literal>(
<replaceable>x
</replaceable>),(
<replaceable>y
</replaceable>)
</literal></entry>
64 <entry>A one-dimensional interval starting at
<replaceable>x
</replaceable> and ending at
<replaceable>y
</replaceable> or vice versa; the
69 <entry><literal>[(
<replaceable>x
</replaceable>),(
<replaceable>y
</replaceable>)]
</literal></entry>
70 <entry>Same as above
</entry>
73 <entry><literal>(
<replaceable>x1
</replaceable>,...,
<replaceable>xn
</replaceable>),(
<replaceable>y1
</replaceable>,...,
<replaceable>yn
</replaceable>)
</literal></entry>
74 <entry>An n-dimensional cube represented by a pair of its diagonally
79 <entry><literal>[(
<replaceable>x1
</replaceable>,...,
<replaceable>xn
</replaceable>),(
<replaceable>y1
</replaceable>,...,
<replaceable>yn
</replaceable>)]
</literal></entry>
80 <entry>Same as above
</entry>
87 It does not matter which order the opposite corners of a cube are
88 entered in. The
<type>cube
</type> functions
89 automatically swap values if needed to create a uniform
90 <quote>lower left
— upper right
</quote> internal representation.
91 When the corners coincide,
<type>cube
</type> stores only one corner
92 along with an
<quote>is point
</quote> flag to avoid wasting space.
96 White space is ignored on input, so
97 <literal>[(
<replaceable>x
</replaceable>),(
<replaceable>y
</replaceable>)]
</literal> is the same as
98 <literal>[ (
<replaceable>x
</replaceable> ), (
<replaceable>y
</replaceable> ) ]
</literal>.
102 <sect2 id=
"cube-precision">
103 <title>Precision
</title>
106 Values are stored internally as
64-bit floating point numbers. This means
107 that numbers with more than about
16 significant digits will be truncated.
111 <sect2 id=
"cube-usage">
115 <xref linkend=
"cube-operators-table"/> shows the specialized operators
116 provided for type
<type>cube
</type>.
119 <table id=
"cube-operators-table">
120 <title>Cube Operators
</title>
124 <entry role=
"func_table_entry"><para role=
"func_signature">
135 <entry role=
"func_table_entry"><para role=
"func_signature">
136 <type>cube
</type> <literal>&&</literal> <type>cube
</type>
137 <returnvalue>boolean
</returnvalue>
140 Do the cubes overlap?
145 <entry role=
"func_table_entry"><para role=
"func_signature">
146 <type>cube
</type> <literal>@
></literal> <type>cube
</type>
147 <returnvalue>boolean
</returnvalue>
150 Does the first cube contain the second?
155 <entry role=
"func_table_entry"><para role=
"func_signature">
156 <type>cube
</type> <literal><@
</literal> <type>cube
</type>
157 <returnvalue>boolean
</returnvalue>
160 Is the first cube contained in the second?
165 <entry role=
"func_table_entry"><para role=
"func_signature">
166 <type>cube
</type> <literal>-
></literal> <type>integer
</type>
167 <returnvalue>float8
</returnvalue>
170 Extracts the
<parameter>n
</parameter>-th coordinate of the cube
176 <entry role=
"func_table_entry"><para role=
"func_signature">
177 <type>cube
</type> <literal>~
></literal> <type>integer
</type>
178 <returnvalue>float8
</returnvalue>
181 Extracts the
<parameter>n
</parameter>-th coordinate of the cube,
182 counting in the following way:
<parameter>n
</parameter> =
2
183 *
<parameter>k
</parameter> -
1 means lower bound
184 of
<parameter>k
</parameter>-th dimension,
<parameter>n
</parameter> =
2
185 *
<parameter>k
</parameter> means upper bound of
186 <parameter>k
</parameter>-th dimension. Negative
187 <parameter>n
</parameter> denotes the inverse value of the corresponding
188 positive coordinate. This operator is designed for KNN-GiST support.
193 <entry role=
"func_table_entry"><para role=
"func_signature">
194 <type>cube
</type> <literal><-
></literal> <type>cube
</type>
195 <returnvalue>float8
</returnvalue>
198 Computes the Euclidean distance between the two cubes.
203 <entry role=
"func_table_entry"><para role=
"func_signature">
204 <type>cube
</type> <literal><#
></literal> <type>cube
</type>
205 <returnvalue>float8
</returnvalue>
208 Computes the taxicab (L-
1 metric) distance between the two cubes.
213 <entry role=
"func_table_entry"><para role=
"func_signature">
214 <type>cube
</type> <literal><=
></literal> <type>cube
</type>
215 <returnvalue>float8
</returnvalue>
218 Computes the Chebyshev (L-inf metric) distance between the two cubes.
226 In addition to the above operators, the usual comparison
227 operators shown in
<xref linkend=
"functions-comparison-op-table"/> are
228 available for type
<type>cube
</type>. These
229 operators first compare the first coordinates, and if those are equal,
230 compare the second coordinates, etc. They exist mainly to support the
231 b-tree index operator class for
<type>cube
</type>, which can be useful for
232 example if you would like a UNIQUE constraint on a
<type>cube
</type> column.
233 Otherwise, this ordering is not of much practical use.
237 The
<filename>cube
</filename> module also provides a GiST index operator class for
238 <type>cube
</type> values.
239 A
<type>cube
</type> GiST index can be used to search for values using the
240 <literal>=
</literal>,
<literal>&&</literal>,
<literal>@
></literal>, and
241 <literal><@
</literal> operators in
<literal>WHERE
</literal> clauses.
245 In addition, a
<type>cube
</type> GiST index can be used to find nearest
246 neighbors using the metric operators
247 <literal><-
></literal>,
<literal><#
></literal>, and
248 <literal><=
></literal> in
<literal>ORDER BY
</literal> clauses.
249 For example, the nearest neighbor of the
3-D point (
0.5,
0.5,
0.5)
250 could be found efficiently with:
252 SELECT c FROM test ORDER BY c
<-
> cube(array[
0.5,
0.5,
0.5]) LIMIT
1;
257 The
<literal>~
></literal> operator can also be used in this way to
258 efficiently retrieve the first few values sorted by a selected coordinate.
259 For example, to get the first few cubes ordered by the first coordinate
260 (lower left corner) ascending one could use the following query:
262 SELECT c FROM test ORDER BY c ~
> 1 LIMIT
5;
264 And to get
2-D cubes ordered by the first coordinate of the upper right
267 SELECT c FROM test ORDER BY c ~
> 3 DESC LIMIT
5;
272 <xref linkend=
"cube-functions-table"/> shows the available functions.
275 <table id=
"cube-functions-table">
276 <title>Cube Functions
</title>
280 <entry role=
"func_table_entry"><para role=
"func_signature">
294 <entry role=
"func_table_entry"><para role=
"func_signature">
295 <function>cube
</function> (
<type>float8
</type> )
296 <returnvalue>cube
</returnvalue>
299 Makes a one dimensional cube with both coordinates the same.
302 <literal>cube(
1)
</literal>
303 <returnvalue>(
1)
</returnvalue>
308 <entry role=
"func_table_entry"><para role=
"func_signature">
309 <function>cube
</function> (
<type>float8
</type>,
<type>float8
</type> )
310 <returnvalue>cube
</returnvalue>
313 Makes a one dimensional cube.
316 <literal>cube(
1,
2)
</literal>
317 <returnvalue>(
1),(
2)
</returnvalue>
322 <entry role=
"func_table_entry"><para role=
"func_signature">
323 <function>cube
</function> (
<type>float8[]
</type> )
324 <returnvalue>cube
</returnvalue>
327 Makes a zero-volume cube using the coordinates defined by the array.
330 <literal>cube(ARRAY[
1,
2,
3])
</literal>
331 <returnvalue>(
1,
2,
3)
</returnvalue>
336 <entry role=
"func_table_entry"><para role=
"func_signature">
337 <function>cube
</function> (
<type>float8[]
</type>,
<type>float8[]
</type> )
338 <returnvalue>cube
</returnvalue>
341 Makes a cube with upper right and lower left coordinates as defined by
342 the two arrays, which must be of the same length.
345 <literal>cube(ARRAY[
1,
2], ARRAY[
3,
4])
</literal>
346 <returnvalue>(
1,
2),(
3,
4)
</returnvalue>
351 <entry role=
"func_table_entry"><para role=
"func_signature">
352 <function>cube
</function> (
<type>cube
</type>,
<type>float8
</type> )
353 <returnvalue>cube
</returnvalue>
356 Makes a new cube by adding a dimension on to an existing cube,
357 with the same values for both endpoints of the new coordinate. This
358 is useful for building cubes piece by piece from calculated values.
361 <literal>cube('(
1,
2),(
3,
4)'::cube,
5)
</literal>
362 <returnvalue>(
1,
2,
5),(
3,
4,
5)
</returnvalue>
367 <entry role=
"func_table_entry"><para role=
"func_signature">
368 <function>cube
</function> (
<type>cube
</type>,
<type>float8
</type>,
<type>float8
</type> )
369 <returnvalue>cube
</returnvalue>
372 Makes a new cube by adding a dimension on to an existing cube. This is
373 useful for building cubes piece by piece from calculated values.
376 <literal>cube('(
1,
2),(
3,
4)'::cube,
5,
6)
</literal>
377 <returnvalue>(
1,
2,
5),(
3,
4,
6)
</returnvalue>
382 <entry role=
"func_table_entry"><para role=
"func_signature">
383 <function>cube_dim
</function> (
<type>cube
</type> )
384 <returnvalue>integer
</returnvalue>
387 Returns the number of dimensions of the cube.
390 <literal>cube_dim('(
1,
2),(
3,
4)')
</literal>
391 <returnvalue>2</returnvalue>
396 <entry role=
"func_table_entry"><para role=
"func_signature">
397 <function>cube_ll_coord
</function> (
<type>cube
</type>,
<type>integer
</type> )
398 <returnvalue>float8
</returnvalue>
401 Returns the
<parameter>n
</parameter>-th coordinate value for the lower
402 left corner of the cube.
405 <literal>cube_ll_coord('(
1,
2),(
3,
4)',
2)
</literal>
406 <returnvalue>2</returnvalue>
411 <entry role=
"func_table_entry"><para role=
"func_signature">
412 <function>cube_ur_coord
</function> (
<type>cube
</type>,
<type>integer
</type> )
413 <returnvalue>float8
</returnvalue>
416 Returns the
<parameter>n
</parameter>-th coordinate value for the
417 upper right corner of the cube.
420 <literal>cube_ur_coord('(
1,
2),(
3,
4)',
2)
</literal>
421 <returnvalue>4</returnvalue>
426 <entry role=
"func_table_entry"><para role=
"func_signature">
427 <function>cube_is_point
</function> (
<type>cube
</type> )
428 <returnvalue>boolean
</returnvalue>
431 Returns true if the cube is a point, that is,
432 the two defining corners are the same.
435 <literal>cube_is_point(cube(
1,
1))
</literal>
436 <returnvalue>t
</returnvalue>
441 <entry role=
"func_table_entry"><para role=
"func_signature">
442 <function>cube_distance
</function> (
<type>cube
</type>,
<type>cube
</type> )
443 <returnvalue>float8
</returnvalue>
446 Returns the distance between two cubes. If both
447 cubes are points, this is the normal distance function.
450 <literal>cube_distance('(
1,
2)', '(
3,
4)')
</literal>
451 <returnvalue>2.8284271247461903</returnvalue>
456 <entry role=
"func_table_entry"><para role=
"func_signature">
457 <function>cube_subset
</function> (
<type>cube
</type>,
<type>integer[]
</type> )
458 <returnvalue>cube
</returnvalue>
461 Makes a new cube from an existing cube, using a list of
462 dimension indexes from an array. Can be used to extract the endpoints
463 of a single dimension, or to drop dimensions, or to reorder them as
467 <literal>cube_subset(cube('(
1,
3,
5),(
6,
7,
8)'), ARRAY[
2])
</literal>
468 <returnvalue>(
3),(
7)
</returnvalue>
471 <literal>cube_subset(cube('(
1,
3,
5),(
6,
7,
8)'), ARRAY[
3,
2,
1,
1])
</literal>
472 <returnvalue>(
5,
3,
1,
1),(
8,
7,
6,
6)
</returnvalue>
477 <entry role=
"func_table_entry"><para role=
"func_signature">
478 <function>cube_union
</function> (
<type>cube
</type>,
<type>cube
</type> )
479 <returnvalue>cube
</returnvalue>
482 Produces the union of two cubes.
485 <literal>cube_union('(
1,
2)', '(
3,
4)')
</literal>
486 <returnvalue>(
1,
2),(
3,
4)
</returnvalue>
491 <entry role=
"func_table_entry"><para role=
"func_signature">
492 <function>cube_inter
</function> (
<type>cube
</type>,
<type>cube
</type> )
493 <returnvalue>cube
</returnvalue>
496 Produces the intersection of two cubes.
499 <literal>cube_inter('(
1,
2)', '(
3,
4)')
</literal>
500 <returnvalue>(
3,
4),(
1,
2)
</returnvalue>
505 <entry role=
"func_table_entry"><para role=
"func_signature">
506 <function>cube_enlarge
</function> (
<parameter>c
</parameter> <type>cube
</type>,
<parameter>r
</parameter> <type>double
</type>,
<parameter>n
</parameter> <type>integer
</type> )
507 <returnvalue>cube
</returnvalue>
510 Increases the size of the cube by the specified
511 radius
<parameter>r
</parameter> in at least
<parameter>n
</parameter>
512 dimensions. If the radius is negative the cube is shrunk instead.
513 All defined dimensions are changed by the
514 radius
<parameter>r
</parameter>. Lower-left coordinates are decreased
515 by
<parameter>r
</parameter> and upper-right coordinates are increased
516 by
<parameter>r
</parameter>. If a lower-left coordinate is increased
517 to more than the corresponding upper-right coordinate (this can only
518 happen when
<parameter>r
</parameter> < 0) than both coordinates are
519 set to their average. If
<parameter>n
</parameter> is greater than the
520 number of defined dimensions and the cube is being enlarged
521 (
<parameter>r
</parameter> > 0), then extra dimensions are added to
522 make
<parameter>n
</parameter> altogether;
0 is used as the initial
523 value for the extra coordinates. This function is useful for creating
524 bounding boxes around a point for searching for nearby points.
527 <literal>cube_enlarge('(
1,
2),(
3,
4)',
0.5,
3)
</literal>
528 <returnvalue>(
0.5,
1.5, -
0.5),(
3.5,
4.5,
0.5)
</returnvalue>
536 <sect2 id=
"cube-defaults">
537 <title>Defaults
</title>
543 select cube_union('(
0,
5,
2),(
2,
3,
1)', '
0');
551 does not contradict common sense, neither does the intersection:
555 select cube_inter('(
0,-
1),(
1,
1)', '(-
2),(
2)');
563 In all binary operations on differently-dimensioned cubes,
564 the lower-dimensional one is assumed to be a Cartesian projection, i. e., having zeroes
565 in place of coordinates omitted in the string representation. The above
566 examples are equivalent to:
570 cube_union('(
0,
5,
2),(
2,
3,
1)','(
0,
0,
0),(
0,
0,
0)');
571 cube_inter('(
0,-
1),(
1,
1)','(-
2,
0),(
2,
0)');
575 The following containment predicate uses the point syntax,
576 while in fact the second argument is internally represented by a box.
577 This syntax makes it unnecessary to define a separate point type
578 and functions for (box,point) predicates.
582 select cube_contains('(
0,
0),(
1,
1)', '
0.5,
0.5');
590 <sect2 id=
"cube-notes">
594 For examples of usage, see the regression test
<filename>sql/cube.sql
</filename>.
598 To make it harder for people to break things, there
599 is a limit of
100 on the number of dimensions of cubes. This is set
600 in
<filename>cubedata.h
</filename> if you need something bigger.
604 <sect2 id=
"cube-credits">
605 <title>Credits
</title>
608 Original author: Gene Selkov, Jr.
<email>selkovjr@mcs.anl.gov
</email>,
609 Mathematics and Computer Science Division, Argonne National Laboratory.
613 My thanks are primarily to Prof. Joe Hellerstein
614 (
<ulink url=
"https://dsf.berkeley.edu/jmh/"></ulink>) for elucidating the
615 gist of the GiST (
<ulink url=
"http://gist.cs.berkeley.edu/"></ulink>), and
616 to his former student Andy Dong for his example written for Illustra.
617 I am also grateful to all Postgres developers, present and past, for
618 enabling myself to create my own world and live undisturbed in it. And I
619 would like to acknowledge my gratitude to Argonne Lab and to the
620 U.S. Department of Energy for the years of faithful support of my database
625 Minor updates to this package were made by Bruno Wolff III
626 <email>bruno@wolff.to
</email> in August/September of
2002. These include
627 changing the precision from single precision to double precision and adding
632 Additional updates were made by Joshua Reich
<email>josh@root.net
</email> in
633 July
2006. These include
<literal>cube(float8[], float8[])
</literal> and
634 cleaning up the code to use the V1 call protocol instead of the deprecated