在PostgreSQL中将bytea转换为双精度

Kei*_*ith 6 sql postgresql double type-conversion bytea

我有一个数据库,其中一个表存储bytea从另一个系统收集的各种通用数据的blob().该bytea字段可以包含任何内容.为了解如何解释数据,该表还具有格式字段.我写了一个Java应用程序读取bytea从数据库的字段byte[],然后我可以很容易地将其转换为double[]int[]或任何格式字段通过说ByteBuffer和各种视图(DoubleBuffer,IntBuffer,等).

现在,我需要在触发器函数中对数据库本身的数据进行一些操作,以便与另一个表保持完整性.我可以找到几乎任何可以想象的数据类型的转换,但我找不到任何从bytea(甚至bit)到double precision和返回的内容.A bytea可以被分解,转换为比特,然后转换为intbigint,但不是a double precision.例如,x'deadbeefdeadbeef'::bit(64)::bigint将转换为-2401053088876216593没有问题,但x'deadbeefdeadbeef'::bit(64)::double precision失败的"错误:无法将类型位转换为双精度"而不是给出IEEE 754的答案-1.1885959257070704E148.

我找到了这个答案/sf/answers/816329461/,它基本上实现了IEEE标准将位转换为double,但是在PostgreSQL中真的没有基本的转换函数吗?另外,我需要向后退double precision,bytea当我完成操作数据并需要更新表时,这个答案没有提供.

有任何想法吗?

Kei*_*ith 2

好吧,我找到了答案。在 PostgreSQL 中,您可以使用 Python 编写函数。为了能够使用 Python,您必须安装 PostgreSQL 安装所需的特定 Python 版本,并将其添加到 PATH 环境变量中。您可以通过查看安装说明来找到您安装的 PostgreSQL 需要哪个版本的 Python。我目前在 Windows 上使用 PostgreSQL 9.6.5,它需要 Python 3.3。我最初尝试了最新的Python 3.6,但它不起作用。我选择了最新的 Windows 版 Python 3.3,即 3.3.5。

安装 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$;
Run Code Online (Sandbox Code Playgroud)

就我而言,所有双精度数都以小端存储,因此我使用<. 我还将模块的导入缓存struct在全局字典中,如/sf/answers/1051779781/中所述。我使用 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');
Run Code Online (Sandbox Code Playgroud)

我得到的答案是

bytea_to_double_array    | encode
double precision[]       | text
-------------------------+------------------
{-1.18859592570707e+148} | efbeaddeefbeadde
Run Code Online (Sandbox Code Playgroud)

哪里'efbeaddeefbeadde''deadbeefdeadbeef'小端字节序。