现在,我需要在触发器函数中对数据库本身的数据进行一些操作,以便与另一个表保持完整性.我可以找到几乎任何可以想象的数据类型的转换,但是我找不到从bytea(或偶数位)到双精度和返回的任何东西. bytea可以被分解,转换为位,然后转换为int或bigint,但不是双精度.例如,x’deadbeefdeadbeef’:: bit(64):: bigint将转换为-2401053088876216593没有问题,但是x’deadbeefdeadbeef’:: bit(64):: double precision失败并出现“ERROR:无法将类型转换为双精度“而不是给出IEEE的754答案-1.1885959257070704E148.
我找到了这个答案https://stackoverflow.com/a/11661849/5274457,它基本上实现了IEEE标准,将位转换为double,但是在Postgresql中真的没有基本的转换函数吗?另外,当我完成操作数据并需要更新表时,我还需要从双精度到字节向后,这个答案没有提供.
有任何想法吗?
安装Python后,通过执行CREATE EXTENSION plpython3u在Postgresql中启用它;在您的数据库中,如此处所述https://www.postgresql.org/docs/current/static/plpython.html.从那里,您可以使用Python主体编写任何函数.
对于我从bytea转换为double precision []并返回的特定情况,我编写了以下函数:
CREATE FUNCTION bytea_to_double_array(b bytea) RETURNS double precision[] LANGUAGE 'plpython3u' AS $BODY$ if 'struct' in GD: struct = GD['struct'] else: import struct GD['struct'] = struct return struct.unpack('<' + str(int(len(b) / 8)) + 'd',b) $BODY$; CREATE FUNCTION double_array_to_bytea(dblarray double precision[]) RETURNS bytea LANGUAGE 'plpython3u' AS $BODY$ if 'struct' in GD: struct = GD['struct'] else: import struct GD['struct'] = struct # dblarray here is really a list. # Postgresql passes sql arrays as Python lists return struct.pack('<' + str(int(len(dblarray))) + 'd',*dblarray) $BODY$;
在我的情况下,所有的双打都存储在小端,所以我使用<.我还在https://stackoverflow.com/a/15025425/5274457中描述了在全局字典中缓存struct模块的导入.我使用GD而不是SD,因为我想在我可能编写的其他函数中使用导入.有关GD和SD的信息,请参阅https://www.postgresql.org/docs/current/static/plpython-sharing.html.
要知道我的数据库中的blob存储为小端,要查看它的运行情况,
SELECT bytea_to_double_array(decode('efbeaddeefbeadde','hex')),encode(double_array_to_bytea(array[-1.1885959257070704E148]),'hex');
我得到的答案是
bytea_to_double_array | encode double precision[] | text -------------------------+------------------ {-1.18859592570707e+148} | efbeaddeefbeadde
其中’efbeaddeefbeadde’是小端的’deadbeefdeadbeef’.
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。