返回首页

Jmix 如何将外部数据直接显示在界面?

来源:博客园

企业级应用中,通常一个业务系统并不是孤立存在的,而是需要与企业、部门或者是外部的已有系统进行集成。一般而言,系统集成的数据和接口交互方式通常有以下几种:

文件传输:通过文件传输的方式将数据传递给其他系统,例如使用 FTP 或 SFTP 等协议传送文件。这种交互方式适合单向或批量数据传输。Web 服务 API:使用 API 与其他系统进行通信,一般采用 SOAP 或 REST 等通信协议。这种交互方式可用于实现单向或双向数据传输。数据库交互:使用数据库共享的方式,或者通过数据库连接进行数据交换。这种交互方式在双向数据同步和实时数据交换方面非常有效。但是一定要注意数据权限控制和数据安全。消息队列:通过消息队列进行数据的收发,而无需与其他系统直接通信。消息队列可用于处理大量数据以及异步数据传输的情况。

不论使用那种方式进行通信和传输,在主系统(即,正在实施的系统)中,还需要考虑的一个问题是,第三方系统过来的数据需不需要保存?如果主系统要基于接入的数据进行进一步处理,则通常需要保存数据。而有时候由于数据安全方面的原因,亦或是考虑到本地存储数据后还存在数据同步与重复存储的问题,第三方系统的数据过来后,主系统并不需要存储数据,只是提供展示和操作界面。

本文中,我们以常见的 REST API 通信为例,看看 Jmix 应用是如何直接使用外部数据的(这里我们不存储外部数据)。


(资料图片)

外部数据源

我们假设外部数据源通过 REST API 提供关于项目(project)和任务(task)的 CRUD 接口。

定义 DTO 和 Service

首先,我们在主系统中定义两个 DTO 实体:ProjectTask,用 Jmix Studio 可以直接创建 DTO 实体:

然后,在主系统中我们需要定义两个 Services,专门用来对 ProjectTask实体进行 CRUD 操作,而这些操作里面,其实是调用了外部系统提供的 REST 接口,以 TaskService为例:

@Componentpublic class TaskService {    public static final String TASKS_BASE_URL = "http://localhost:18080/tasks";    @Autowired    private RestTemplate restTemplate;    public List 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 response = restTemplate.postForEntity(url, task, Task.class);        return response.getBody();    }    public void deleteTask(Task task) {        restTemplate.delete(TASKS_BASE_URL + "/" + task.getId());    }}

ProjectDTO 的创建过程和 ProjectService的内容与上面步骤类似,这里就不再赘述。

第一种方式:使用代理

第一种方式是使用数据加载代理和提交代理方法,将原本使用 DataManager进行数据加载和写入的相应方法替换为使用我们自定义的服务:

这里,我们选择 TaskDTO 和它的列表页和编辑页作为示例。

首先,在列表页添加数据加载的代理,在界面选中数据加载器后,双击代理方法中的 标签,Studio 会自动生成方法并跳转到方法定义,添加自定义逻辑:

然后,在编辑页添加数据提交代理,这里需要在 XML 中选中 data节点,然后双击生成 commitDelegate

这样就完成了我们需要实现的功能。是不是很简单?

第二种方式:自定义数据存储

Jmix 中,数据存储可以进行自定义,通过自定义的数据存储,可以像处理 JPA 实体一样,使用 DataManager处理 DTO 实体。在检测到 DTO 实体关联到某个自定义存储后,DataManager会将 CRUD 操作都通过代理执行,并且能处理对 DTO 实体的引用。具体实现框架如下:

可以看到,使用这种方式不需要对界面中的实体操作进行拦截,而是将所有对于外部系统的接口调用都交给 DataManager通过数据存储进行分发。

这里,我们选择 ProjectDTO 作为示例,创建相应的数据存储:

创建 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_jmix.core.store-descriptor_projectds = sample_ProjectDataStoreDescriptor
Project实体添加 @Store注解:
@Store(name = "projectds")@JmixEntitypublic class Project {    ...}

通过这几步,我们完成了数据存储的实现和配置。Project的列表页和编辑也不需要做任何改动,并且,任何通过 DataManagerProjectDTO 的操作就像操作 JPA 实体一样方便,可以在服务层和 UI 层调用。

结论

如果外部 API 提供了丰富的操作接口,比如 CRUD、分页、排序甚至支持某种查询语言,那么我们推荐创建一个自定义的数据存储。这种为数据操作 Service 提供自定义数据存储的方式,更贴近 Jmix 原生的开发方式。另外,如果需要的话,自定义的数据存储也可以继承 AbstractDataStore类,这个类是 Jmix 内置 JpaDataStore的父类。通过这个类派生可以使用框架提供的一些机制,比如数据访问安全和对外部数据的审计。

但是如果外部 API 只提供了几个简单的接口,这种情况我们建议直接在 UI 层使用数据读写代理的方式。

示例的完整代码请访问 GitHub。

标签: