云数据库 GAUSSDB-范围类型:定义新的范围类型
定义新的范围类型
用户可以定义自己的范围类型。这样做最常见的原因是为了使用内建范围类型中提供的subtype上没有的范围。例如,要创建一个subtype float8的范围类型:
gaussdb=# CREATE TYPE floatrange AS RANGE ( subtype = float8, subtype_diff = float8mi ); gaussdb=# SELECT '[1.234, 5.678]'::floatrange; floatrange --------------- [1.234,5.678] (1 row) gaussdb=# DROP TYPE floatrange;
因为float8没有有意义的“步长”,在这个例子中没有定义一个正规化函数。
定义自己的范围类型也允许用户指定使用一个不同的子类型B-树操作符类或者集合, 以便更改排序顺序来决定哪些值会落入到给定的范围中。
如果subtype被认为是具有离散值而不是连续值,CREATE TYPE命令应当指定一个canonical函数。正规化函数接收一个输入的范围值,并且必须返回一个可能具有不同界限和格式的等价的范围值。对于两个表示相同值集合的范围(例如[1, 7]和[1, 8)),正规的输出必须相同。选择哪一种表达作为正规的没有关系,只要两个具有不同格式的等价值总是能被映射到具有相同格式的相同值就行。除了调整包含/排除界限格式外,假设期望的补偿比subtype能够存储的要大,一个正规化函数可能会舍入边界值。例如,一个timestamp之上的范围类型可能被定义为具有一个一小时的步长,这样正规化函数可能需要对不是一小时的倍数的界限进行舍入,或者可能直接抛出一个错误。
subtype差异函数采用两个subtype输入值,并且返回表示为一个float8值的差(即X减Y)。在上面的例子中,可以使用常规float8减法操作符之下的函数。但是对于任何其他subtype,可能需要某种类型转换。还可能需要一些关于如何把差异表达为数字的创新型想法。为了最大的可扩展性,subtype_diff函数应该同意选中的操作符类和排序规则所蕴含的排序顺序,即只要它的第一个参数根据排序顺序大于第二个参数,它的结果就应该是正值。
subtype_diff函数相关示例:
gaussdb=# CREATE FUNCTION time_subtype_diff(x time, y time) RETURNS float8 AS 'SELECT EXTRACT(EPOCH FROM (x - y))' LANGUAGE sql STRICT IMMUTABLE; gaussdb=# CREATE TYPE timerange AS RANGE ( subtype = time, subtype_diff = time_subtype_diff ); gaussdb=# SELECT '[11:10, 23:00]'::timerange; timerange --------------------- [11:10:00,23:00:00] (1 row) gaussdb=# DROP TYPE timerange; gaussdb=# DROP FUNCTION time_subtype_diff;
更多关于创建范围类型的信息请参考CREATE TYPE。