Jmix 如何将外部数据直接显示在界面?
企业级应用中,通常一个业务系统并不是孤立存在的,而是需要与企业、部门或者是外部的已有系统进行集成。一般而言,系统集成的数据和接口交互方式通常有以下几种:
- 文件传输:通过文件传输的方式将数据传递给其他系统,例如使用 FTP 或 SFTP 等协议传送文件。这种交互方式适合单向或批量数据传输。
- Web 服务 API:使用 API 与其他系统进行通信,一般采用 SOAP 或 REST 等通信协议。这种交互方式可用于实现单向或双向数据传输。
- 数据库交互:使用数据库共享的方式,或者通过数据库连接进行数据交换。这种交互方式在双向数据同步和实时数据交换方面非常有效。但是一定要注意数据权限控制和数据安全。
- 消息队列:通过消息队列进行数据的收发,而无需与其他系统直接通信。消息队列可用于处理大量数据以及异步数据传输的情况。
不论使用那种方式进行通信和传输,在主系统(即,正在实施的系统)中,还需要考虑的一个问题是,第三方系统过来的数据需不需要保存?如果主系统要基于接入的数据进行进一步处理,则通常需要保存数据。而有时候由于数据安全方面的原因,亦或是考虑到本地存储数据后还存在数据同步与重复存储的问题,第三方系统的数据过来后,主系统并不需要存储数据,只是提供展示和操作界面。
本文中,我们以常见的 REST API 通信为例,看看 Jmix 应用是如何直接使用外部数据的(这里我们不存储外部数据)。
外部数据源
我们假设外部数据源通过 REST API 提供关于项目(project)和任务(task)的 CRUD 接口。
定义 DTO 和 Service
首先,我们在主系统中定义两个 DTO 实体:Project
和 Task
,用 Jmix Studio 可以直接创建 DTO 实体:
然后,在主系统中我们需要定义两个 Services,专门用来对 Project
和 Task
实体进行 CRUD 操作,而这些操作里面,其实是调用了外部系统提供的 REST 接口,以 TaskService
为例:
@Component
public class TaskService {
public static final String TASKS_BASE_URL = "http://localhost:18080/tasks";
@Autowired
private RestTemplate restTemplate;
public List<Task> loadTasks() {
Task[] tasks = restTemplate.getForObject(TASKS_BASE_URL, Task[].class);
return Arrays.asList(tasks);
}
public Task saveTask(Task task) {
String url = task.getId() != null ?
TASKS_BASE_URL + "/" + task.getId() :
TASKS_BASE_URL;
ResponseEntity<Task> response = restTemplate.postForEntity(url, task, Task.class);
return response.getBody();
}
public void deleteTask(Task task) {
restTemplate.delete(TASKS_BASE_URL + "/" + task.getId());
}
}
Project
DTO 的创建过程和 ProjectService
的内容与上面步骤类似,这里就不再赘述。
第一种方式:使用代理
第一种方式是使用数据加载代理和提交代理方法,将原本使用 DataManager
进行数据加载和写入的相应方法替换为使用我们自定义的服务:
这里,我们选择 Task
DTO 和它的列表页和编辑页作为示例。
首先,在列表页添加数据加载的代理,在界面选中数据加载器后,双击代理方法中的 <empty>
标签,Studio 会自动生成方法并跳转到方法定义,添加自定义逻辑:
然后,在编辑页添加数据提交代理,这里需要在 XML 中选中 data
节点,然后双击生成 commitDelegate
:
这样就完成了我们需要实现的功能。是不是很简单?
第二种方式:自定义数据存储
Jmix 中,数据存储可以进行自定义,通过自定义的数据存储,可以像处理 JPA 实体一样,使用 DataManager
处理 DTO 实体。在检测到 DTO 实体关联到某个自定义存储后,DataManager
会将 CRUD 操作都通过代理执行,并且能处理对 DTO 实体的引用。具体实现框架如下:
可以看到,使用这种方式不需要对界面中的实体操作进行拦截,而是将所有对于外部系统的接口调用都交给 DataManager
通过数据存储进行分发。
这里,我们选择 Project
DTO 作为示例,创建相应的数据存储:
- 创建
ProjectDataStore
实现DataStore
接口:
@Component("sample_ProjectDataStore")
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class ProjectDataStore implements DataStore {
// 注入 ProjectService 用于具体的操作
@Autowired
private ProjectService projectService;
// 后面需要实现接口中的方法,主要是通过 projectService 对数据进行 CRUD,这里省略。
...
}
- 创建一个实现
StoreDescriptor
接口的类。必须是一个 Spring 单例 bean,其中getBeanName()
方法必须返回上一步创建的 bean 的名称:
@Component("sample_ProjectDataStoreDescriptor")
public class ProjectDataStoreDescriptor implements StoreDescriptor {
@Override
public String getBeanName() {
return "sample_ProjectDataStore";
}
@Override
public boolean isJpa() {
return false;
}
}
- 在
application.properties
中添加对数据存储的配置:
# 如果有多个,则以逗号分隔
jmix.core.additional-stores = projectds
# 配置名称为 jmix.core.storeDescriptor_<store_name>
jmix.core.store-descriptor_projectds = sample_ProjectDataStoreDescriptor
- 为
Project
实体添加@Store
注解:
@Store(name = "projectds")
@JmixEntity
public class Project {
...
}
通过这几步,我们完成了数据存储的实现和配置。Project
的列表页和编辑也不需要做任何改动,并且,任何通过 DataManager
对 Project
DTO 的操作就像操作 JPA 实体一样方便,可以在服务层和 UI 层调用。
结论
如果外部 API 提供了丰富的操作接口,比如 CRUD、分页、排序甚至支持某种查询语言,那么我们推荐创建一个自定义的数据存储。这种为数据操作 Service 提供自定义数据存储的方式,更贴近 Jmix 原生的开发方式。另外,如果需要的话,自定义的数据存储也可以继承 AbstractDataStore
类,这个类是 Jmix 内置 JpaDataStore
的父类。通过这个类派生可以使用框架提供的一些机制,比如数据访问安全和对外部数据的审计。
但是如果外部 API 只提供了几个简单的接口,这种情况我们建议直接在 UI 层使用数据读写代理的方式。
示例的完整代码请访问 GitHub。