PingKai Logo下载

使用 Hibernate 连接到 TiDB

TiDB 是一个兼容 MySQL 的数据库。Hibernate 是当前比较流行的开源 Java 应用持久层框架。由于 TiDB 与 MySQL 高度兼容,建议使用 org.hibernate.dialect.MySQLDialect 作为 Hibernate 的方言 (Dialect),以获得更好的长期兼容性。或者,也可以使用 TiDB 特定的方言 (org.hibernate.community.dialect.TiDBDialect),该方言位于 Hibernate community dialects 项目中,但该项目并非由 PingCAP 维护。如果你在使用 MySQLDialect 时遇到兼容性问题,可以在 GitHub 上提交 issue

本文档将展示如何使用 TiDB 和 Hibernate 来完成以下任务:

  • 配置你的环境。
  • 使用 Hibernate 连接到 TiDB。
  • 构建并运行你的应用程序。你也可以参考示例代码片段,完成基本的 CRUD 操作。

前置需求

运行代码并连接到 TiDB

本小节演示如何运行示例应用程序的代码,并连接到 TiDB。

第 1 步:克隆示例代码仓库到本地

运行以下命令,将示例代码仓库克隆到本地:

git clone https://github.com/tidb-samples/tidb-java-hibernate-quickstart.git
cd tidb-java-hibernate-quickstart

第 2 步:配置连接信息

根据不同的 TiDB 部署方式,使用不同的方法连接到 TiDB。

  1. 在 TiDB Cloud 的 My TiDB 页面中,选择你的 TiDB Cloud Starter 或 Essential 实例,进入实例的 Overview 页面。

  2. 点击右上角的 Connect 按钮,将会弹出连接对话框。

  3. 确认对话框中的配置和你的运行环境一致。

    • Connection TypePublic
    • Branch 选择 main
    • Connect With 选择 General
    • Operating System 为你的运行环境。
  4. 如果你还没有设置密码,点击 Generate Password 生成一个随机密码。

  5. 运行以下命令,将 env.sh.example 复制并重命名为 env.sh

    cp env.sh.example env.sh
  6. 复制并粘贴对应连接字符串至 env.sh 中。需更改部分示例结果如下:

    export TIDB_HOST='{host}'  # e.g. gateway01.ap-northeast-1.prod.aws.tidbcloud.com
    export TIDB_PORT='4000'
    export TIDB_USER='{user}'  # e.g. xxxxxx.root
    export TIDB_PASSWORD='{password}'
    export TIDB_DB_NAME='test'
    export USE_SSL='true'

    注意替换 {} 中的占位符为连接对话框中获得的值。

    TiDB Cloud Starter 要求使用 TLS (SSL) connection,因此 USE_SSL 的值应为 true

  7. 保存 env.sh 文件。

第 3 步:运行代码并查看结果

  1. 运行下述命令,执行示例代码:

    make
  2. 查看 Expected-Output.txt,并与你的程序输出进行比较。结果近似即为连接成功。

示例代码片段

你可参考以下关键代码片段,完成自己的应用开发。

完整代码及其运行方式,见代码仓库 tidb-java-hibernate-quickstart

连接到 TiDB

编写配置文件 hibernate.cfg.xml

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <session-factory>

        
        <property name="hibernate.connection.driver_class">com.mysql.cj.jdbc.Driver</property>
        <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
        <property name="hibernate.connection.url">${tidb_jdbc_url}</property>
        <property name="hibernate.connection.username">${tidb_user}</property>
        <property name="hibernate.connection.password">${tidb_password}</property>
        <property name="hibernate.connection.autocommit">false</property>

        
        <property name="hibernate.hbm2ddl.auto">create-drop</property>

        
        <property name="hibernate.show_sql">true</property>
        <property name="hibernate.format_sql">true</property>
    </session-factory>
</hibernate-configuration>

请将 ${tidb_jdbc_url}${tidb_user}${tidb_password} 等替换为你的 TiDB 的实际值。随后编写以下函数:

public SessionFactory getSessionFactory() {
    return new Configuration()
            .configure("hibernate.cfg.xml")
            .addAnnotatedClass(${your_entity_class})
            .buildSessionFactory();
}

在使用该函数时,你需要替换 ${your_entity_class} 为自己的数据实体类。如果你有多个实体类,需要添加多个 .addAnnotatedClass(${your_entity_class}) 语句。此外,这仅是 Hibernate 的其中一种配置方式。在配置中遇到任何问题,或想了解更多关于 Hibernate 的信息,你可参考 Hibernate 官方文档

插入或更新数据

try (Session session = sessionFactory.openSession()) {
    session.persist(new PlayerBean("id", 1, 1));
}

更多信息参考插入数据更新数据

查询数据

try (Session session = sessionFactory.openSession()) {
    PlayerBean player = session.get(PlayerBean.class, "id");
    System.out.println(player);
}

更多信息参考查询数据

删除数据

try (Session session = sessionFactory.openSession()) {
    session.remove(new PlayerBean("id", 1, 1));
}

更多信息参考删除数据

MySQLDialect 的兼容性

在 TiDB 中使用 MySQLDialect 时,请注意以下行为:

SERIALIZABLE 隔离级别

如果应用尝试将事务隔离级别设置为 SERIALIZABLE,TiDB 会返回如下错误:

The isolation level 'SERIALIZABLE' is not supported. Set tidb_skip_isolation_level_check=1 to skip this error

要避免该错误,请在服务端设置以下 TiDB 系统变量:

SET GLOBAL tidb_skip_isolation_level_check=1;

启用该变量后,TiDB 会接受请求中指定 SERIALIZABLE 而不返回错误。TiDB 内部仍然使用 REPEATABLE-READ,这是 TiDB 所支持的最强事务隔离级别。更多信息,请参见 tidb_skip_isolation_level_check

CHECK 约束

Hibernate 的 @Check 注解会生成 DDL CHECK 约束。MySQL 8.0.16 及之后版本 默认会强制执行这些约束。但在 TiDB 中,如果没有显式启用,则不会强制执行。

要在 TiDB 中启用 CHECK 约束的强制执行,请设置以下系统变量:

SET GLOBAL tidb_enable_check_constraint=ON;

如果未启用该设置,TiDB 会接受 CHECK 约束的语法但不会强制执行,这可能导致数据完整性问题。更多信息,请参见 CHECK 约束

下一步

需要帮助?