使用 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 的 Clusters 页面中,选择你的 TiDB Cloud Starter 集群,进入集群的 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 文件。

    1. 在 TiDB Cloud 的 Clusters 页面中,选择你的 TiDB Cloud Dedicated 集群,进入集群的 Overview 页面。

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

    3. 在连接对话框中,从 Connection Type 下拉列表中选择 Public,并点击 CA cert 下载 CA 文件。

      如果你尚未配置 IP 访问列表,请在首次连接前点击 Configure IP Access List 或按照配置 IP 访问列表(英文)中的步骤进行配置。

      Public 连接类型外,TiDB Cloud Dedicated 还支持 Private EndpointVPC Peering 连接类型。详情请参阅连接 TiDB Cloud Dedicated 集群(英文)

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

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

      export TIDB_HOST='{host}' # e.g. tidb.xxxx.clusters.tidb-cloud.com export TIDB_PORT='4000' export TIDB_USER='{user}' # e.g. root export TIDB_PASSWORD='{password}' export TIDB_DB_NAME='test' export USE_SSL='false'

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

    6. 保存 env.sh 文件。

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

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

      export TIDB_HOST='{host}' export TIDB_PORT='4000' export TIDB_USER='root' export TIDB_PASSWORD='{password}' export TIDB_DB_NAME='test' export USE_SSL='false'

      注意替换 {} 中的占位符为你的 TiDB 对应的值,并设置 USE_SSLfalse。如果你在本机运行 TiDB,默认 Host 地址为 127.0.0.1,密码为空。

    3. 保存 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> <!-- Database connection settings --> <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> <!-- Required so a table can be created from the 'PlayerDAO' class --> <property name="hibernate.hbm2ddl.auto">create-drop</property> <!-- Optional: Show SQL output for debugging --> <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 约束

    下一步

    需要帮助?

    如果在开发的过程中遇到问题,可以在 AskTUG 上进行提问,寻求帮助。

    文档内容是否有帮助?