【问题标题】:Read MySQL binary(16) UUID with Java使用 Java 读取 MySQL 二进制 (16) UUID
【发布时间】:2013-07-18 14:44:42
【问题描述】:

这应该是一个非常简单的问题,我只是在这里遗漏了一些基本的东西,而我正在经历“那些日子……” 不能使用 Hibernate 或其他 ORM。使用 Java PreparedStatement。

MySQL 的东西:

CREATE TABLE `article` (
  `articleID` binary(16) NOT NULL,
  `publisherID` bigint(20) DEFAULT NULL,
  PRIMARY KEY (`articleID`),
) ENGINE=InnoDB DEFAULT CHARSET=utf8$$

insert into article ( articleID, publisherID )
values ( (UNHEX(REPLACE(UUID(),'-',''))), 1111  );

Java 的东西

PreparedStatement ps = connection.prepareStatement( "select articleID, publisherID from article" );
ResultSet rs = ps.executeQuery();
while( rs.next())
{
  byte[] artIDArr = rs.getBytes( "articleID" );
  UUID artID = UUID.nameUUIDFromBytes( artIDArr );
}

rs.close();
ps.close();

现在,从数据库中读取 UUID...

    select hex(articleID) from article;

1C711C50E4773873AB1533401E2F420C
A1FCD341EE9311E297B700FFB00BB509
A95E06B6EEE611E297B700FFB00BB509

但是将我在 java 代码中读到的内容转出:

6c825dc9-c98f-37ab-b01b-416294811a84
de6337f9-f276-3e30-b9a3-8d9338a1977f
57ccb5af-1a66-329f-b069-69638e1af24f

现在,这是因为我在将它们存储为二进制之前从 UUID 中删除了破折号,并且补液假设它们在那里?

在 MySql 中将存储为二进制 (16) 的 UUID 读取到 Jav UUID 对象的正确方法是什么?

编辑:如果我将preparedStatment查询更改为“选择hex(articleID)作为articleID ...”并将其作为字符串读取,它当然是数据库包含的内容,但UUID会抛出异常,因为缺少字符串破折号...

【问题讨论】:

  • 您确定将这些存储为二进制文件值得吗?节省的长度通常很少,而且手动查询非常烦人。
  • 是的,我们将有一个巴西的记录,我们需要它是一个更小的 PK。是的,UUID 并不是最好的 PK。我知道。但这就是我所拥有的。谢谢。

标签: java mysql uuid


【解决方案1】:
UUID artID = UUID.nameUUIDFromBytes(artIDArr);

使用 MD5 并修补字节。使用类似的东西

static UUID toUUID(byte[] bytes) {
    if (bytes.length != 16) {
        throw new IllegalArgumentException();
    }
    int i = 0;
    long msl = 0;
    for (; i < 8; i++) {
        msl = (msl << 8) | (bytes[i] & 0xFF);
    }
    long lsl = 0;
    for (; i < 16; i++) {
        lsl = (lsl << 8) | (bytes[i] & 0xFF);
    }
    return new UUID(msl, lsl);
}
UUID artID = toUUID(artIDArr);

【讨论】:

  • Java 7 jdk 不会使 UUID(byte[]) 构造函数可见。“构造函数 UUID(byte[]) 不可见”
  • @MonkeyWrench 我确实监督过;为它制作了一个函数,并更正了答案。
  • 哈哈。也许我应该将破折号存储在数据库中,在 select 语句中使用“hex()”,然后在 Java 中将它们作为字符串读取。但我很惊讶没有更好的方法来处理这个问题。 “每个人”使用 binary(16) 以 MySQL 状态存储 UUID 是可行的方法。我在这里错过了什么?
【解决方案2】:

所以我通过在查询中使用 hex() 函数调用包装 binary(16) 来解决这个问题。不确定哪个更有效,让数据库处理转换或翻转 Java 中的位。无论哪种方式,完成。

  PreparedStatement ps = connection.prepareStatement( "select hex(articleID) as articleID, publisherID from article" );

  ResultSet rs = ps.executeQuery();
  while( rs.next())
  {
    String artIDStr = rs.getString( "articleID" );
    UUID artID = getUUIDFromString( artIDStr );

我会将 Eggen 的答案标记为正确,因为他付出了努力,而且它可能会奏效。 8)

【讨论】:

  • 如果您不受性能问题的限制,这是一个很好的解决方案。将 16 字节数字解析为字符串,然后将生成的字符串解析回 16 字节数字(这是 UUID 内部的内容)是一种为性能改进敞开大门的解决方案。
猜你喜欢
  • 1970-01-01
  • 2021-12-19
  • 1970-01-01
  • 2021-11-05
  • 2021-07-26
  • 2011-07-25
  • 1970-01-01
  • 1970-01-01
  • 2015-03-06
相关资源
最近更新 更多