我有一个数据库,其中一个表存储从另一个系统收集的各种通用数据的blob(bytea
)。 bytea
字段中可以包含任何内容。为了解如何解释数据,该表还具有格式字段。我编写了一个Java应用程序,以bytea
的形式从数据库中读取byte[]
字段,然后我可以轻松地将其转换为double[]
或int[]
或格式字段所指的任何内容使用ByteBuffer
和各种观点(DoubleBuffer
,IntBuffer
等)。
现在,我需要在触发器函数中对数据库本身的数据进行一些操作,以便与另一个表保持完整性。我可以找到几乎可以想象的任何数据类型的转换,但是我找不到从bytea
(或甚至bit
)到double precision
并返回的任何内容。可以将bytea
拆分,转换为位,然后转换为int
或bigint
,而不是double precision
。例如,x'deadbeefdeadbeef'::bit(64)::bigint
将毫无问题地转换为-2401053088876216593
,但x'deadbeefdeadbeef'::bit(64)::double precision
失败并显示“错误:无法将类型位转换为双倍精度”而不是{75}的IEEE 754答案}}
我找到了这个答案https://stackoverflow.com/a/11661849/5274457,它基本上实现了IEEE标准,将位转换为double,但是在PostgreSQL中真的没有基本的转换函数吗?另外,当我完成操作数据并需要更新表时,我需要从-1.1885959257070704E148
向后退double precision
,这个答案没有提供。
有什么想法吗?
答案 0 :(得分:1)
好的,我找到了答案。在PostgreSQL中,您可以使用Python编写函数。为了能够使用Python,您必须安装PostgreSQL安装所需的特定版本的Python,并在PATH环境变量中使用它。通过查看安装说明,您可以找到PostgreSQL安装所需的Python版本。我目前在Windows上使用PostgreSQL 9.6.5,它需要Python 3.3。我最初尝试了最新的Python 3.6,但它不起作用。我决定使用最新的Python 3.3 for Windows,即3.3.5。
安装Python之后,通过在数据库上执行bytea
来在PostgreSQL中启用它,如https://www.postgresql.org/docs/current/static/plpython.html所述。从那里,您可以使用Python主体编写任何函数。
对于我从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$;
并返回的特定情况,我编写了以下函数:
<
在我的情况下,所有双打都存储在小端,所以我使用struct
。我还在https://stackoverflow.com/a/15025425/5274457中描述的全局字典中缓存SELECT bytea_to_double_array(decode('efbeaddeefbeadde', 'hex')), encode(double_array_to_bytea(array[-1.1885959257070704E148]), 'hex');
模块的导入。我使用GD而不是SD,因为我想在我可能编写的其他函数中使用导入。有关GD和SD的信息,请参阅https://www.postgresql.org/docs/current/static/plpython-sharing.html。
要知道我的数据库中的blob存储为小端,
,请查看它的运行情况bytea_to_double_array | encode
double precision[] | text
-------------------------+------------------
{-1.18859592570707e+148} | efbeaddeefbeadde
我得到的答案是
'efbeaddeefbeadde'
其中'deadbeefdeadbeef'
是小端的{{1}}。