使用Java读取MySQL二进制(16) UUID

6
这应该是一个非常简单的问题,我只是在某些基础方面遗漏了一些东西,而且今天可能不顺利...不能使用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正确地读取为Java UUID对象的方法是什么?

编辑:如果我将preparedStatment查询更改为"select hex(articleID) as articleID..."并将其作为字符串读取,则它显然是DB包含的内容,但UUID会抛出异常,因为该字符串缺少破折号...


你确定将它们存储为二进制值是值得的吗?长度节省通常很小,而且手动查询会变得非常麻烦。 - tadman
是的,我们将有数十亿条记录,并且需要它成为较小的主键。是的,UUID不是最好的主键。我知道。但这就是我所拥有的。谢谢。 - MonkeyWrench
2个回答

5
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
@MonkeyWrench 我确实疏忽了;我为此编写了一个函数,并纠正了答案。 - Joop Eggen
2
也许我应该将破折号存储在数据库中,在选择语句中使用“hex()”,并在Java中将它们读取为字符串。但是我很惊讶没有更好的方法来处理这个问题。“每个人”都将UUID存储在MySQL状态中,使用binary(16)是正确的方法。我在这里错过了什么? - MonkeyWrench

5

所以,我通过在查询中使用hex()函数调用来简单地包装二进制(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)


1
这是一个不错的解决方案,如果你不受性能限制的话。将一个16字节的数字解析成字符串,然后再将生成的字符串解析回一个16字节的数字(这就是UUID在内部的表示方式)是一种留下性能优化空间的解决方案。 - scottb

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接