0

0

Jakarta EE JPA 容器管理实体管理器与内存数据库配置指南

心靈之曲

心靈之曲

发布时间:2025-11-27 16:08:22

|

801人浏览过

|

来源于php中文网

原创

Jakarta EE JPA 容器管理实体管理器与内存数据库配置指南

本教程详细阐述了在 jakarta ee 环境中,如何利用 `@datasourcedefinition` 注解配置容器管理的 jpa 实体管理器以使用内存数据库(如 hsqldb)。文章涵盖了 `persistence.xml` 的 jta 数据源设置、`@datasourcedefinition` 的使用方法及注意事项,旨在帮助开发者实现便捷的测试环境部署和事务管理。

容器管理 JPA 实体管理器概述

在 Jakarta EE 应用中,容器管理的 JPA 实体管理器(Container-Managed EntityManager)是推荐的实践方式。它通过依赖注入(@PersistenceContext)提供 EntityManager 实例,并与容器的 JTA(Java Transaction API)事务管理器集成,允许开发者使用声明式事务(@Transactional)来简化事务管理,无需手动处理事务的开始、提交和回滚。

要实现容器管理的实体管理器,persistence.xml 文件中的持久化单元必须声明 transaction-type="JTA",并且需要通过 <jta-data-source> 元素引用一个 JTA 数据源。这个数据源通常通过 JNDI (Java Naming and Directory Interface) 名称在应用服务器中进行查找。

配置内存数据库的数据源

当使用内存数据库(如 HSQLDB)进行开发或测试时,我们希望能够方便地定义和引用数据源,而无需依赖复杂的服务器特定配置。Jakarta EE 提供了 @DataSourceDefinition 注解,允许在应用程序代码中以编程方式定义数据源,并使其通过 JNDI 可用。

使用 @DataSourceDefinition 定义数据源

@DataSourceDefinition 注解可以放置在一个普通的 Java 类上,通常是一个空的类,其主要目的是承载此注解。以下是如何为 HSQLDB 内存数据库定义数据源的示例:

import jakarta.annotation.sql.DataSourceDefinition;

/**
 * HSQLDB内存数据库的数据源定义。
 * 此类仅用于承载@DataSourceDefinition注解,本身无需任何方法或字段。
 */
@DataSourceDefinition(
    class = "org.hsqldb.jdbcDriver", // JDBC驱动类名
    name = "java:app/jdbc/testdb", // JNDI名称,供persistence.xml引用
    url = "jdbc:hsqldb:mem:testdb;DB_CLOSE_DELAY=-1", // HSQLDB内存数据库URL
    user = "sa", // 数据库用户名
    password = "" // 数据库密码
)
public class HsqldbDataSourceConfig {
    // 无需任何代码
}

注解属性说明:

  • class: 指定 JDBC 驱动的完全限定类名。对于 HSQLDB,通常是 org.hsqldb.jdbcDriver。
  • name: 这是数据源将在 JNDI 注册的名称。persistence.xml 中的 <jta-data-source> 将使用此名称进行查找。建议使用 java:app/jdbc/YourDataSourceName 或 java:global/jdbc/YourDataSourceName 这样的 JNDI 命名约定。java:app 表示数据源在当前应用程序范围内可见。
  • url: 数据库的连接 URL。对于 HSQLDB 内存数据库,jdbc:hsqldb:mem:testdb 定义了一个名为 testdb 的内存数据库。DB_CLOSE_DELAY=-1 是 HSQLDB 特有的参数,表示在最后一个连接关闭后,数据库实例不会立即关闭,这对于测试非常有用,可以保持数据库状态。
  • user: 连接数据库的用户名。
  • password: 连接数据库的密码。

当应用程序部署到支持 Jakarta EE 的应用服务器时,容器会扫描并处理 @DataSourceDefinition 注解,将指定的数据源注册到 JNDI 目录中。

更新 persistence.xml 配置

一旦数据源通过 @DataSourceDefinition 定义并注册到 JNDI,persistence.xml 文件就需要引用这个 JNDI 名称,并且必须移除所有直接的 JDBC 连接属性,因为这些属性现在由 JNDI 数据源本身提供。

GentleAI
GentleAI

GentleAI是一个高效的AI工作平台,为普通人提供智能计算、简单易用的界面和专业技术支持。让人工智能服务每一个人。

下载

修改后的 persistence.xml 示例如下:

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.2"
             xmlns="http://xmlns.jcp.org/xml/ns/persistence"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_2.xsd">

    <persistence-unit name="test" transaction-type="JTA">
        <!-- 引用通过@DataSourceDefinition定义的JNDI数据源名称 -->
        <jta-data-source>java:app/jdbc/testdb</jta-data-source>
        <class>demo.Jakarta.user.UserEntity</class> <!-- 您的实体类 -->
        <properties>
            <!-- 数据库模式生成策略,这里设置为创建 -->
            <property name="jakarta.persistence.schema-generation.database.action" value="create"/>
            <!-- 移除所有jakarta.persistence.jdbc.*相关的属性 -->
        </properties>
    </persistence-unit>
</persistence>

关键点:

  • <jta-data-source> 元素的值必须与 @DataSourceDefinition 中 name 属性的值完全匹配。
  • jakarta.persistence.jdbc.driver、jakarta.persistence.jdbc.url、jakarta.persistence.jdbc.user 和 jakarta.persistence.jdbc.password 等属性不再需要,并且应该从 persistence.xml 中移除。这些信息现在由 JNDI 数据源提供。

使用容器管理的实体管理器和事务

配置完成后,您就可以在您的业务逻辑(如 Repository 或 Service 层)中注入容器管理的 EntityManager,并利用 @Transactional 注解进行事务管理。

import jakarta.persistence.EntityManager;
import jakarta.persistence.PersistenceContext;
import jakarta.transaction.Transactional;
import demo.Jakarta.user.UserEntity; // 假设您的实体类

/**
 * 用户数据访问层示例
 */
public class UserRepository {

    // 注入容器管理的EntityManager
    @PersistenceContext(unitName = "test")
    private EntityManager em;

    /**
     * 保存用户实体,由容器管理事务。
     * @param user 待保存的用户实体
     * @return 保存后的用户实体
     */
    @Transactional
    public UserEntity save(UserEntity user) {
        em.persist(user); // 持久化操作
        return user;
    }

    /**
     * 根据ID查找用户实体。
     * @param id 用户ID
     * @return 对应的用户实体,如果不存在则为null
     */
    public UserEntity findById(Long id) {
        return em.find(UserEntity.class, id);
    }
}

在上述示例中,@PersistenceContext(unitName = "test") 会注入一个与名为 "test" 的持久化单元关联的 EntityManager 实例。@Transactional 注解确保 save 方法在一个 JTA 事务中执行,容器会自动管理事务的生命周期。

注意事项与最佳实践

  1. 可移植性考量: 将数据源配置硬编码到应用程序的 @DataSourceDefinition 中,虽然方便,但会降低应用程序在不同环境(如开发、测试、生产)之间部署时的灵活性。在生产环境中,通常推荐通过应用服务器的管理控制台或特定配置文件来定义数据源,实现配置与代码的分离。这样,无需重新编译和打包应用程序即可更改数据源配置。
  2. JNDI 命名约定: 选择合适的 JNDI 名称。java:app/jdbc/ 作用域表示数据源在当前应用程序内可见,而 java:global/jdbc/ 则表示在整个应用服务器实例中可见。根据您的需求选择合适的命名空间。
  3. 事务管理器: 确保您的 Jakarta EE 应用服务器已正确配置 JTA 事务管理器,以便 @Transactional 注解能够正常工作。
  4. 开发测试优势: @DataSourceDefinition 策略特别适用于开发和测试阶段,尤其是在使用嵌入式或内存数据库时。它允许开发者快速启动和运行应用程序,无需复杂的服务器配置步骤,从而提高开发效率。

总结

通过 @DataSourceDefinition 注解,Jakarta EE 提供了一种便捷的方式来在应用程序内部定义和注册 JTA 数据源,尤其适用于使用内存数据库的开发和测试场景。结合 persistence.xml 中的 jta-data-source 配置,开发者可以轻松地利用容器管理的 JPA 实体管理器和声明式事务。尽管这种方法在开发阶段具有显著优势,但在生产环境中,仍需权衡其可移植性,并考虑采用服务器级别的配置方式。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1949

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2119

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1171

2024.11.28

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

891

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.06

go中interface用法
go中interface用法

本专题整合了go语言中int相关内容,阅读专题下面的文章了解更多详细内容。

78

2025.09.10

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

389

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2111

2023.08.14

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 81.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号