【问题标题】:How to use Hibernate Annotations to add an index on a Lob / Clob / tinyblob如何使用 Hibernate Annotations 在 Lob / Clob / tinyblob 上添加索引
【发布时间】:2009-05-13 14:36:42
【问题描述】:

我有一个使用 Hibernate 映射到数据库的 bean。我正在使用 Hibernate Annotations 来指示我想要的映射,并创建索引。彻底简化的代码如下所示。

我遇到的问题是我的 byte[] 字段上的索引没有创建;特别是我的多字段索引 sysUuid 没有被创建(参见示例代码)。在 Hibernate 调试日志中,我什至没有看到创建索引的尝试!

我想指出,uuid 字段上的@Index 注释也不会导致数据库上的索引。

我知道如何使用 MySQL 手动创建索引:

create index sysuuid on persons ( system, `uuid`(8) );

有趣的特点是需要转义 uuid(因为它是 MySQL 函数)并且需要在字段上给出长度(就像文本字段一样)。

但是,我还没有找到使用 Hibernate Annotations 提供索引长度字段的方法,因此我无法测试这是否是问题所在。但是可以肯定的是,在注解中命名字段“uuid(8)”是行不通的。

@Entity
// The UniqueConstraints work 
@Table(name = "persons", 
   uniqueConstraints = {@UniqueConstraint(columnNames = {"uid", "system"}) } )
// but these don't generate an index
@org.hibernate.annotations.Table(appliesTo="persons", 
   indexes={@Index(name="sysUuid",  columnNames={"system", "uuid"})  } )
public class Person  {
    @Basic 
    @NotNull
    private String uid;

    @Basic
    private int system;

    // Gets mapped to tinyblob
    @Basic
    @Size(min = 16, max = 16)
    private byte[] uuid;

    // getters and setters here 
}

我想问你的是:是否可以使用注释在 lob 上添加索引,如果可以,如何?

编辑

我确实有可能转向基于字符串的 UUID,但我对此并不满意,因为 uuid 在概念上是一个 16 字节的标识符。

我非常喜欢 Java 类型来匹配问题域。

正如我所说 - 我确实有一个方便的 SQL 语句,因此我可以部署代码 + 一个 SQL 脚本。我只是认为在可行的情况下最好使用自记录代码。

编辑和添加赏金

我相信我需要的索引不能使用 Hibernate Annotations 创建(回复。Matt Solnit 的回答)。

但是,我希望了解更多有关使用 Hibernate Annotations 创建索引的信息,因此最终的答案最终会记录 API。

【问题讨论】:

    标签: java mysql hibernate


    【解决方案1】:

    您可以使用 Hibernate 的 auxiliary objects 支持来做到这一点,但不能使用注释来做到这一点:-(。

    在您的示例中,它看起来像这样(为简洁起见,省略了很多内容):

    <class name="Person" table="persons">
      <!-- whatever -->
      <database-object>
        <create>create index sysuuid on persons ( system, `uuid`(8) )</create>
        <drop>drop index sysuuid</drop>
        <dialect-scope name="org.hibernate.dialect.MySQL5InnoDBDialect" />
      </database-object>
    </class>
    

    对于缺少基于注释的答案,我深表歉意 :-(。希望这会有所帮助。

    注意:如果您确实采用这种方法,请注意方言范围必须完全匹配。例如,如果您的 Hibernate 配置要求使用 MySQL5InnoDBDialect,那么您也必须在 &lt;dialect-scope&gt; 元素中包含此方言。使用 MySQLDialect 将不起作用,即使它是 InnoDB 方言的超类。

    【讨论】:

    • 谢谢。我不会使用这种方法,而是坚持使用我的 SQL 脚本。如果我有一些 Hibernate 专家作为同事,我可能会采用这个解决方案。但是我担心(由于您的注释)将来在切换方言时可能会出现问题:) 对于我的公司而言,SQL 脚本是一种更直观的解决方案。
    • 有谁知道是否有办法在主键上指定索引长度?
    【解决方案2】:

    您真正想要做的是将您的 UUID 属性定义为字符串而不是字节数组。这样,Hibernate 会将其映射到数据库中的字符列而不是 LOB,我认为它会创建您想要的索引。

    【讨论】:

    • 感谢您的回答。但是一个 byte[] 是我所拥有的。我对根据我的一个工具的功能修改我的问题域并不十分满意。我的继任者不明白为什么我的 UUID(来自外部来源的字节 [])突然变成了字符串。除此之外 - String 应该是哪个字符集? :-) 现在我只是用我的应用程序提供一个索引创建 SQL 脚本。
    【解决方案3】:

    这是我对 MySQL 的研究结果。如果你有一个长度很大的字段:

    @Column(length = 2000)
    public String getMessageId() {
        return messageId;
    }
    

    那么它的类型将是 BLOB 并且索引将不起作用:

    HHH000389: Unsuccessful: create index messageid_idx on MessageDetails (messageId)
    BLOB/TEXT column 'messageId' used in key specification without a key length      
    

    如果使用注解,添加自定义sql的唯一方法如下:

        cfg.addAuxiliaryDatabaseObject(new SimpleAuxiliaryDatabaseObject(
                "CREATE INDEX messageId_IDX ON MessageDetails(MessageID(128))",
                "DROP INDEX messageId_IDX ON MessageDetails"));
    

    其中 cfg 是用于注册注解的对象:

        cfg.addAnnotatedClass(MessageDetails.class);
    

    这对我有用。感谢您的关注。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-05-31
      • 1970-01-01
      • 2019-02-27
      • 1970-01-01
      • 2016-02-20
      • 1970-01-01
      相关资源
      最近更新 更多