openGauss数据类型(5)

范围类型

范围类型是表达某种元素类型(称为范围的_subtype_)的一个值的范围的数据类型。例如,timestamp的范围可以被用来表达一个会议室被保留的时间范围。在这种情况下,数据类型是tsrange(“timestamp range”的简写)而timestamp是 subtype。subtype 必须具有一种总体的顺序,这样对于元素值是在一个范围值之内、之前或之后就是界线清楚的。

范围类型非常有用,因为它们可以表达一种单一范围值中的多个元素值,并且可以很清晰地表达诸如范围重叠等概念。用于时间安排的时间和日期范围是最清晰的例子;但是价格范围、一种仪器的量程等等也都有用。

  • 内建范围类型

    有下列内建范围类型:

    • int4range — integer的范围
    • int8range — bigint的范围
    • numrange — numeric的范围
    • tsrange — 不带时区的 timestamp的范围
    • tstzrange — 带时区的 timestamp的范围
    • daterange — date的范围
  • 包含和排除边界

    每一个非空范围都有两个界限,下界和上界。上下界之间的所有值都被包括在范围内。一个包含界限意味着边界点本身也被包括在范围内,而一个排除边界意味着边界点不被包括在范围内。

    在一个范围的文本形式中,一个包含下界被表达为“[”而一个排除下界被表达为“(”。同样,一个包含上界被表达为“]”而一个排除上界被表达为“)”。

    函数lower_inc和upper_inc分别测试一个范围值的上下界。

  • 无限(无界)范围

    一个范围的下界可以被忽略,意味着所有小于上界的值都被包括在范围中,例如(,3]。 同样,如果范围的上界被忽略,那么所有比上界大的值都被包括在范围中。如果上下界都被忽略,该元素类型的所有值都被认为在该范围中。 规定缺失的包括界限自动转换为排除,例如,[,] 转换为 (,)。 你可以认为这些缺失值为 +/- 无穷大,但它们是特殊范围类型值,并且被视为超出任何范围元素类型的 +/- 无穷大值。

    具有“infinity”概念的元素类型可以用它们作为显式边界值。例如,在时间戳范围,[today,infinity)不包括特殊的timestamp值infinity,尽管 [today,infinity] 包括它,就好比 [today,) 和 [today,]。

    函数lower_inf和upper_inf分别测试一个范围的无限上下界。

  • 范围输入输出

    一个范围值的输入必须遵循下列模式之一:

    (lower-bound,upper-bound) 
    (lower-bound,upper-bound] 
    [lower-bound,upper-bound) 
    [lower-bound,upper-bound] 
    empty 
    

    圆括号或方括号指示上下界是否为排除的或者包含的。注意最后一个模式是empty,它表示一个空范围(一个不包含点的范围)。

    _lower-bound_可以是作为subtype的合法输入的一个字符串,或者是空表示没有下界。同样,_upper-bound_可以是作为 subtype 的合法输入的一个字符串,或者是空表示没有上界。

    每个界限值可以使用"(双引号)字符引用。如果界限值包含圆括号、方括号、逗号、双引号或反斜线时,这样做是必须的,因为否则那些字符会被认作范围语法的一部分。要把一个双引号或反斜线放在一个被引用的界限值中,就在它前面放一个反斜线(还有,在一个双引号引用的界限值中的一对双引号表示一个双引号字符,这与 SQL 字符串中的单引号规则类似)。此外,你可以避免引用并且使用反斜线转义来保护所有数据字符,否则它们会被当做返回语法的一部分。还有,要写一个是空字符串的界限值,则可以写成"",因为什么都不写表示一个无限界限。

    范围值前后允许有空格,但是圆括号或方括号之间的任何空格会被当做上下界值的一部分(取决于元素类型,它可能是也可能不是有意义的)。

    例子:

    -- 包括 3,不包括 7,并且包括 3 和 7 之间的所有点 
    SELECT '[3,7)'::int4range;  
    -- 既不包括 3 也不包括 7,但是包括之间的所有点 
    SELECT '(3,7)'::int4range;  
    -- 只包括单独一个点 4 
    SELECT '[4,4]'::int4range;  
    -- 不包括点(并且将被标准化为 '空') 
    SELECT '[4,4)'::int4range; 
    
  • 构造范围

    每一种范围类型都有一个与其同名的构造器函数。使用构造器函数常常比写一个范围文字常数更方便,因为它避免了对界限值的额外引用。构造器函数接受两个或三个参数。两个参数的形式以标准的形式构造一个范围(下界是包含的,上界是排除的),而三个参数的形式按照第三个参数指定的界限形式构造一个范围。第三个参数必须是下列字符串之一: “()”、 “(]”、 “[)”或者 “[]”。 例如:

    -- 完整形式是:下界、上界以及指示界限包含性/排除性的文本参数。 
    SELECT numrange(1.0, 14.0, '(]');  
    -- 如果第三个参数被忽略,则假定为 '[)'。 
    SELECT numrange(1.0, 14.0);  
    -- 尽管这里指定了 '(]',显示时该值将被转换成标准形式,因为 int8range 是一种离散范围类型(见下文)。 
    SELECT int8range(1, 14, '(]');  
    -- 为一个界限使用 NULL 导致范围在那一边是无界的。 
    SELECT numrange(NULL, 2.2); 
    
  • 离散范围类型

    一种范围的元素类型具有一个良定义的“步长”,例如integer或date。在这些类型中,如果两个元素之间没有合法值,它们可以被说成是相邻。这与连续范围相反,连续范围中总是(或者几乎总是)可以在两个给定值之间标识其他元素值。例如,numeric类型之上的一个范围就是连续的,timestamp上的范围也是(尽管timestamp具有有限的精度,并且在理论上可以被当做离散的,最好认为它是连续的,因为通常并不关心它的步长)。

    另一种考虑离散范围类型的方法是对每一个元素值都有一种清晰的“下一个”或“上一个”值。了解了这种思想之后,通过选择原来给定的下一个或上一个元素值来取代它,就可以在一个范围界限的包含和排除表达之间转换。例如,在一个整数范围类型中,[4,8]和(3,9)表示相同的值集合,但是对于 numeric 上的范围就不是这样。

    一个离散范围类型应该具有一个_正规化_函数,它知道元素类型期望的步长。正规化函数负责把范围类型的相等值转换成具有相同的表达,特别是与包含或者排除界限一致。如果没有指定一个正规化函数,那么具有不同格式的范围将总是会被当作不等,即使它们实际上是表达相同的一组值。

    内建的范围类型int4range、int8range和daterange都使用一种正规的形式,该形式包括下界并且排除上界,也就是[)。不过,用户定义的范围类型可以使用其他习惯。

  • 定义新的范围类型

    用户可以定义他们自己的范围类型。这样做最常见的原因是为了使用内建范围类型中没有提供的 subtype 上的范围。例如,要创建一个 subtype float8的范围类型:

    CREATE TYPE floatrange AS RANGE (
         subtype = float8,
         subtype_diff = float8mi 
    );
    SELECT '[1.234, 5.678]'::floatrange; 
    

    因为float8没有有意义的“步长”,我们在这个例子中没有定义一个正规化函数。

    定义自己的范围类型也允许你指定使用一个不同的子类型 B-树操作符类或者集合, 以便更改排序顺序来决定哪些值会落入到给定的范围中。

    如果 subtype 被认为是具有离散值而不是连续值,CREATE TYPE命令应当指定一个canonical函数。正规化函数接收一个输入的范围值,并且必须返回一个可能具有不同界限和格式的等价的范围值。对于两个表示相同值集合的范围(例如[1, 7]和[1, 8)),正规的输出必须一样。选择哪一种表达作为正规的没有关系,只要两个具有不同格式的等价值总是能被映射到具有相同格式的相同值就行。除了调整包含/排除界限格式外,假使期望的补偿比 subtype 能够存储的要大,一个正规化函数可能会舍入边界值。例如,一个timestamp之上的范围类型可能被定义为具有一个一小时的步长,这样正规化函数可能需要对不是一小时的倍数的界限进行舍入,或者可能直接抛出一个错误。

    另外,任何打算要和 GiST 或 SP-GiST 索引一起使用的范围类型应当定一个 subtype 差异或subtype_diff函数(没有subtype_diff时索引仍然能工作,但是可能效率不如提供了差异函数时高)。subtype 差异函数采用两个 subtype 输入值,并且返回表示为一个float8值的差(即_X_减_Y_)。在我们上面的例子中,可以使用常规float8减法操作符之下的函数。但是对于任何其他 subtype,可能需要某种类型转换。还可能需要一些关于如何把差异表达为数字的创新型想法。为了最大的可扩展性,subtype_diff函数应该同意选中的操作符类和排序规则所蕴含的排序顺序,也就是说,只要它的第一个参数根据排序顺序大于第二个参数,它的结果就应该是正值。

    subtype_diff函数的一个不那么过度简化的例子:

    CREATE FUNCTION time_subtype_diff(x time, y time) RETURNS float8 AS 'SELECT EXTRACT(EPOCH FROM (x - y))' LANGUAGE sql STRICT IMMUTABLE;  
    CREATE TYPE timerange AS RANGE (     
         subtype = time,     
         subtype_diff = time_subtype_diff
     );  
    SELECT '[11:10, 23:00]'::timerange; 
    

索引

可以为范围类型的表列创建 GiST 和 SP-GiST 索引。例如,要创建一个 GiST 索引:

CREATE INDEX reservation_idx ON reservation USING GIST (during); 

一个 GiST 或 SP-GiST 索引可以加速涉及以下范围操作符的查询: =、 &&、 <@、 @>、 «、 >>、 -|-、 &<以及 &> 。

此外,B-树和哈希索引可以在范围类型的表列上创建。对于这些索引类型,基本上唯一有用的范围操作就是等值。使用相应的< 和 >操作符,对于范围值定义有一种 B-树排序顺序,但是该顺序相当任意并且在真实世界中通常不怎么有用。范围类型的 B-树和哈希支持主要是为了允许在查询内部进行排序和哈希,而不是创建真正的索引。

对象标识符类型

openGauss在内部使用对象标识符(OID)作为各种系统表的主键。系统不会给用户创建的表增加一个OID系统字段,OID类型代表一个对象标识符。

目前OID类型用一个四字节的无符号整数实现。因此不建议在创建的表中使用OID字段做主键。

表 15 对象标识符类型

名称

引用

描述

示例

OID

-

数字化的对象标识符。

564182

CID

-

命令标识符。它是系统字段cmin和cmax的数据类型。命令标识符是32位的量。

-

XID

-

事务标识符。它是系统字段xmin和xmax的数据类型。事务标识符也是64位的量。

-

TID

-

行标识符。它是系统表字段ctid的数据类型。行ID是一对数值(块号,块内的行索引),它标识该行在其所在表内的物理位置。

-

REGCONFIG

pg_ts_config

文本搜索配置。

english

REGDICTIONARY

pg_ts_dict

文本搜索字典。

simple

REGOPER

pg_operator

操作符名。

-

REGOPERATOR

pg_operator

带参数类型的操作符。

*(integer,integer)或-(NONE,integer)

REGPROC

pg_proc

函数名称。

sum

REGPROCEDURE

pg_proc

带参数类型的函数。

sum(int4)

REGCLASS

pg_class

关系名。

pg_type

REGTYPE

pg_type

数据类型名。

integer

OID类型:主要作为数据库系统表中字段使用。

示例:

openGauss=# SELECT oid FROM pg_class WHERE relname = 'pg_type';
 oid  
------
 1247
(1 row)

OID别名类型REGCLASS:主要用于对象OID值的简化查找。

示例:

openGauss=# SELECT attrelid,attname,atttypid,attstattarget FROM pg_attribute WHERE attrelid = 'pg_type'::REGCLASS;
 attrelid |  attname   | atttypid | attstattarget 
----------+------------+----------+---------------
     1247 | xc_node_id     |       23 |             0
     1247 | tableoid       |       26 |             0
     1247 | cmax           |       29 |             0
     1247 | xmax           |       28 |             0
     1247 | cmin           |       29 |             0
     1247 | xmin           |       28 |             0
     1247 | oid            |       26 |             0
     1247 | ctid           |       27 |             0
     1247 | typname        |       19 |            -1
     1247 | typnamespace   |       26 |            -1
     1247 | typowner       |       26 |            -1
     1247 | typlen         |       21 |            -1
     1247 | typbyval       |       16 |            -1
     1247 | typtype        |       18 |            -1
     1247 | typcategory    |       18 |            -1
     1247 | typispreferred |       16 |            -1
     1247 | typisdefined   |       16 |            -1
     1247 | typdelim       |       18 |            -1
     1247 | typrelid       |       26 |            -1
     1247 | typelem        |       26 |            -1
     1247 | typarray       |       26 |            -1
     1247 | typinput       |       24 |            -1
     1247 | typoutput      |       24 |            -1
     1247 | typreceive     |       24 |            -1
     1247 | typsend        |       24 |            -1
     1247 | typmodin       |       24 |            -1
     1247 | typmodout      |       24 |            -1
     1247 | typanalyze     |       24 |            -1
     1247 | typalign       |       18 |            -1
     1247 | typstorage     |       18 |            -1
     1247 | typnotnull     |       16 |            -1
     1247 | typbasetype    |       26 |            -1
     1247 | typtypmod      |       23 |            -1
     1247 | typndims       |       23 |            -1
     1247 | typcollation   |       26 |            -1
     1247 | typdefaultbin  |      194 |            -1
     1247 | typdefault     |       25 |            -1
     1247 | typacl         |     1034 |            -1
(38 rows)

伪类型

openGauss数据类型中包含一系列特殊用途的类型,这些类型按照类别被称为伪类型。伪类型不能作为字段的数据类型,但是可以用于声明函数的参数或者结果类型。

当一个函数不仅是简单地接受并返回某种SQL数据类型的情况下伪类型是很有用的。表16列出了所有的伪类型。

表 16 伪类型

名称

描述

any

表示函数接受任何输入数据类型。

anyelement

表示函数接受任何数据类型。

anyarray

表示函数接受任意数组数据类型。

anynonarray

表示函数接受任意非数组数据类型。

anyenum

表示函数接受任意枚举数据类型。

anyrange

表示函数接受任意范围数据类型。

cstring

表示函数接受或者返回一个空结尾的C字符串。

internal

表示函数接受或者返回一种服务器内部的数据类型。

language_handler

声明一个过程语言调用句柄返回language_handler。

fdw_handler

声明一个外部数据封装器返回fdw_handler。

record

标识函数返回一个未声明的行类型。

trigger

声明一个触发器函数返回trigger。

void

表示函数不返回数值。

opaque

一个已经过时的类型,以前用于所有上面这些用途。

声明用C编写的函数(不管是内置的还是动态装载的)都可以接受或者返回任何这样的伪数据类型。当伪类型作为参数类型使用时,用户需要保证函数的正常运行。

用过程语言编写的函数只能使用实现语言允许的伪类型。目前,过程语言都不允许使用作为参数类型的伪类型,并且只允许使用void和record作为结果类型。一些多态的函数还支持使用anyelement、anyarray、anynonarray anyenum和anyrange类型。

伪类型internal用于声明那种只能在数据库系统内部调用的函数,他们不能直接在SQL查询里调用。如果函数至少有一个internal类型的参数,则不能从SQL里调用他。建议不要创建任何声明返回internal的函数,除非他至少有一个internal类型的参数。

示例:

--创建表
openGauss=# create table t1 (a int);

--插入两条数据
openGauss=# insert into t1 values(1),(2);

--创建函数showall()。
openGauss=# CREATE OR REPLACE FUNCTION showall() RETURNS SETOF record
AS $$ SELECT count(*) from t1; $$
LANGUAGE SQL;

--调用函数showall()。
openGauss=# SELECT showall();
 showall 
---------
 (2)
(1 row)

--删除函数。
openGauss=# DROP FUNCTION showall();

--删除表
openGauss=# drop table t1;
  • 0
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中的`LocalDateTime`是一个表示日期时间值的对象,它包含了年、月、日以及小时、分钟、秒等信息,但不包含时区信息。这意味着它只关注“是什么”(date and time),而不考虑“在哪里”(timezone)。这使得`LocalDateTime`在处理跨时区的数据时非常有用,因为它不会引入因时区转换而产生的复杂性。 ### Java中的LocalDateTime 在Java 8及更高版本中,`LocalDateTime`是`java.time`包下的一个类,用于存储日期和时间信息。其构造函数允许传入年、月、日、小时、分钟和秒,并提供一系列方法来获取和设置各个部分的时间信息,如: ```java LocalDateTime now = LocalDateTime.now(); System.out.println("Current date and time: " + now); ``` ### OpenGauss对应的类型 OpenGauss是一种开源的关系型数据库系统,它是基于PostgreSQL发展而来。在OpenGauss中,对应于Java中的`LocalDateTime`类型的是`TIMESTAMP WITHOUT TIME ZONE`数据类型。这个数据类型主要用于存储不带时区信息的完整日期和时间值,类似于`LocalDateTime`的概念。 #### TIMESTAMP WITHOUT TIME ZONE in OpenGaussOpenGauss中,`TIMESTAMP WITHOUT TIME ZONE`可以存储日期、时间和时分秒等详细信息,但它并不包括时区信息。这意味着当你从一个不受特定时区影响的应用程序(例如Java应用)读取数据到OpenGauss表中时,你可以使用`TIMESTAMP WITHOUT TIME ZONE`类型,然后在查询结果显示或进一步操作时,需要考虑接收者的本地时区。 #### 示例 如果你在Java应用程序中生成了一个`LocalDateTime`实例并将其插入到OpenGauss数据库数据库会将这个日期时间视作没有时区信息的本地时间。因此,在检索这些数据后,如果要在非默认时区中查看它们,你需要根据接收者所在时区对这些时间进行相应的调整。 #### 关联问题: 1. **Java如何将LocalDateTime转换为OpenGauss的TIMESTAMP WITHOUT TIME ZONE类型**? 2. **在OpenGauss中如何处理TIMESTAMP WITHOUT TIME ZONE类型以适应不同用户所在的时区**? 3. **比较Java中的LocalDateTime与OpenGauss的TIMESTAMP WITHOUT TIME ZONE类型的异同及其应用场景**?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值