简介 对任何企业及应用来说,数据模型都是应用的基石之一。在设计数据模型时,除了需要考虑业务数据的要求之外,还应该思考一些有可能会影响应用程序设计的问题。比如,是否需要数据库表行级。..
一、Jmix数据访问层
1.1 简介
1.2 JPA数据模型
对任何企业及应用来说,数据模型都是应用的基石之一。在设计数据模型时,除了需要考虑业务数据的要求之外,还应该思考一些有可能会影响应用程序设计的问题。比如,是否需要数据库表行级的安全机制?是否需要使用软删除?是否需要 CRUD API,如果需要的话,谁来干这个枯燥的活?
事实上,JPA 已经是 Java 应用程序中创建数据模型的标准了。但是 JPA 不提供能实现高级安全机制的接口,也没有软删除,所以有需求的程序员只能自己。
Jmix 框架提供了通用的引擎,可以为使用 JPA 的应用程序增加额外的功能。在不破坏 JPA 兼容性的情况下,Jmix 框架可以让您愉快的使用高级数据安全机制,无感的使用软删除,并且为 CRUD 操作提供通用 REST API 以及其他的功能。
本文将讨论 Jmix 中的数据访问层,谈谈使用该框架及其工具到底能做什么,还会讨论数据访问层的一些底层原理。
1.3 数据模型,JPA 和 @JmixEntity
对于熟悉 JPA 的 Java 开发者来说,使用 Jmix 不需要学习任何新的知识。只需要创建 JPA 实体即可。Jmix 也提供了可视化设计器可以帮助完成这一工作。
在 Jmix 的可视化设计器中,您需要提供实体名称,选择 ID 类型(UUID、Long、Integer、String 或者嵌入类),然后选择需要支持的功能:
实体版本 - 支持乐观锁
创建时审计 - 增加创建时间、创建人
更新时审计 - 增加更新时间、更新人
软删除 - 增加删除时间、删除人

最后,您会得到类似如下的 JPA 类:
1 |
|
注意,Jmix 并非侵入式的影响你的代码。生成的实体只是添加了一些注解,不需要实现任何框架特有的接口,更不需要继承某个类。这样的非侵入式设计,在实现您应用程序特有的实体关系和架构时,能保持足够的灵活度以适应你的业务需求。
Jmix 使用的大部分注解要么来自 JPA 要么来自 Spring Boot JPA 库:@Entity、@Versioned、@CreatedBy、@LastModifiedBy
,您之前也许都见过。Jmix 框架依赖 Spring Boot 的具体实现来支持乐观锁以及创建和更新审计。
下面看看数据模型中几个 Jmix 特有的注解:
@JmixEntity
- 引导 Jmix 数据访问层将此类添加至实体仓库(Entities repository)
@JmixGeneratedValue
- 表示实体的 ID
属性必须由 Jmix 生成并指定
@DeletedBy
和 @DeletedDate
- 标记字段,当使用软删除方案时用来标记实体的删除状态
您也许会问:“什么是 Jmix 数据访问层中的实体仓库?你们是不是有自己的数据访问引擎?” 答案是也不是。Jmix 框架会存储应用程序数据模型的额外信息 - metamodel(元模型),并且增加了一个特殊的工具 - DataManager
- 用来访问数据,但是其底层还是用的 JPA。
Metamodel 和 DataManager
Jmix 基于 Spring boot 框架构建。Boot 以广泛使用 IoC 模式而闻名天下,该模式意味着将有关 bean 的所有数据存储在特殊的存储库中 - ApplicationContext
。除了这个存储之外,Jmix 将关于应用程序数据模型的所有数据(比如 metadata)存储在另一个存储库中 - Metadata
。
Jmix 中
是数据访问层的核心。包含关于实体、属性、实体关系等等所有的信息。这些信息是在应用程序启动时对类进行扫描的过程中一次收集的,Jmix 用这些信息来实现神奇的功能:从行级数据安全机制到自动生成的管理界面 UI 的双向数据绑定。
为了有效的使用 Jmix 数据访问层,需要使用 DataManager 组件。该组件是 JPA 中著名 EntityManager
的包装类。与
类似,
能使用 JPQL 查询语句或者 ID 加载实体,保存或删除实体以及对所选实例进行计数。
下面是
的一些示例用法。
通过 ID
加载一个实体,使用 Java 类作为参数:
1 | Speaker loadSpeakerById(UUID speakerId) { |
}
`使用 JPQL 查询加载数据:```java
1 | List<Speaker> loadSpeakesByCompanyEmail() { |
}
1 | 使用 |
会分析 JPQL,然后依据策略的定义限制该 JPQL 的执行或修改 JPQL 中的属性列表以符合安全策略。
如需定义行级角色,您需要制定一个 JPQL 策略,该 JQPL 策略由
where以及可选的
join子句组成,用来限制从数据库表中选取的数据:
1 |
|
1 |
|
}因此,带有行级角色用户发起的所有查询都会通过这种方式进行转换:```text where子句会用 AND 操作符添加在现有的条件之后,text join `子句也会添加在现有的之后。如果 JPQL 中没有这两个关键字,相应的子句会直接添加在后面。text
Metadata
1 | 存储库在这里扮演了重要的角色。它能帮助我们正确的解析并修改 JPQL 并能找出受限的属性和实体,无论使用的是 JPQL 还是 Java 类来定义 DB 查询。 |
语句,而所有的
find() 请求都会被修改,添加一个合适的 子句用来过滤那些被 “删除” 的实体。
@Id
@DeletedBy
@Column(name = “DELETED_BY”)
private String deletedBy;
}
`这里我们需要```text
1 | 来找出哪一列需要更新或者添加到 where 子句中。与 CUBA 不同,Jmix 不要求实体实现一个特定的接口才能支持软删除。而只需要将两个注解添加至实的两个字段即可。 |
然后就可以使用类似这样的查询了:
{
speakerCount
1 | speakerList(limit: 10, offset: 0, orderBy: {name: ASC}) { |
id
name
email
}
1 |
|
本文标题: Jmix的数据访问层将
发布时间: 2022年03月01日 00:00
最后更新: 2025年12月30日 08:54
原始链接: https://haoxiang.eu.org/43796a38/
版权声明: 本文著作权归作者所有,均采用CC BY-NC-SA 4.0许可协议,转载请注明出处!

