oop - 我们认为模型与 MVVM(应用于 JavaFX)的关系是什么?
问题描述
我对如何在 JavaFX 中组织代码有点困惑(假设我打算采用 MVVM 方式)。
考虑下图的标准 4 层架构:
MVVM 中的“模型”在哪里?它是在“用户界面”层内还是在“应用程序/域层”上?我会想象 MVVM 的模型是适合当前视图的后端模型的表示(投影),但我不确定我在网上看到的情况。
举个例子:
我有一个简单的冰淇淋应用程序。它的应用层公开了这些方法:
IceCreamService.iceCreams(): List<String>
;IceCreamService.flavours(iceCream: String): List<String>
.
UI 将有一个带有冰淇淋列表的组合框,并且基于用户的选择将允许他选择一些口味。然后有一个按钮来提交订单。
问题是:
我们认为这里的模型是什么?这IceCreamService
是我们所说的模型还是模型是我们在 UI 级别创建的由属性/可观察对象组成的类?如果是后者,谁负责用数据填充这个模型?“控制器”(ViewModel)或者它应该足够聪明来自己做(也就是说,它有一个引用IceCreamService
?)
谢谢!
解决方案
MVC 及其衍生的设计模式总是存在“10 个人会找到 12 种实现方式”的问题,所以我将简要指出一些“流行”的实现方式。
到目前为止,常见的是您的“用户界面”肯定映射到 MVVM 中的“视图”。然后从这里开始有点争议。
ViewModel 是大脑
实现 MVVM 的一种流行方式是“ViewModel”是整个应用程序的大脑。在这种情况下,“Application”映射到“ViewModel”,“Domain”映射到“Model”。ViewModel 控制(并决定)获取数据的内容和方式,以及它想要向 View 公开的内容和方式。大多数时候“模型”只是一个普通的 POJO 类(例如IceCream
类)。
对于 JavaFX,您可以让 Model 和 ViewModel 中的所有属性由某种Property
实现(即实现该接口的东西)表示。如果您没有使用“mvvmFX”之类的 API,那么您的视图很可能会将属性绑定到 ViewModel 中的其他属性。如果您使用的是“mvvmFX”(我没有使用),那么您(很可能)将通过属性名称将 View 属性绑定到 ViewModel(这是 WPF 所做的)。
在这种方法中,ViewModel 负责获取冰淇淋列表。IceCreamService
可以被认为是一个单独的Service
层,或者您可以假设它是 ViewModel 的一部分(它本身不是 ViewModel,但它属于 ViewModel)。如果您想更新此列表,则由 ViewModel 决定。
模型作为域
另一种实现是让模型进行数据管理,很像hibernate在Spring中所做的。模型类继续表示数据(或实体等),但它带有自己的持久性逻辑。这意味着如果您更新模型对象的值(例如更改IceCream
实例的风格),模型会知道它是“脏的”,并将更改推送到存储库。
MVCVM
我还阅读了另一种方法。简而言之,它类似于第一种方法,但是它将IceCreamService
类移动到一个名为“Controller”的全新层中。
你的例子
根据您的示例,这是说明如何在 JavaFX 中实现它的示例。
class IceCreamSelectionView { // The FXML controller class used as MVVM View class
@FXML ComboBox<String> iceCreams;
@FXML ComboBox<String> flavors;
private final IceCreamSelectionViewModel vm; // The corresponding ViewModel
@FXML private void initialize() {
Bindings.bindContent(iceCreams.getItems(), vm.getIceCreams());
Bindings.bindContent(flavors.getItems(), vm.getFlavors());
vm.selectedIceCreamProperty().bind(iceCreams.valueProperty());
vm.selectedFlavorProperty().bind(flavors.valueProperty());
}
}
class IceCreamSelectionViewModel {
/*
* The properties
*/
public final ObservableList<String> getIceCreams() { return iceCreams; }
public final ObservableList<String> getFlavors() { return flavors; }
public final StringProperty selectedIceCreamProperty() { return selectedIceCream; }
public final StringProperty selectedFlavorProperty() { return selectedFlavor; }
private updateIceCreams() {
iceCreams.setAll(iceCreamService.getIceCreams());
}
}
// There is no Model class because both "iceCream" and "flavor" cannot be further broken down.
推荐阅读
- php - 在 PHP MYSQLI 中第二次查询数据库后出现异常未知错误
- r - 在 Shiny 中动态渲染盒子
- ibm-mobilefirst - 通过 REST API 注册推送通知后,设备未在 mfpconsole 中显示
- java - 无法实现斯坦福 CoreNLP 的 OpenIE 注释器的选项
- php - php imagick 脚本以错误 500 结尾
- angularjs - 无法使用 rquirejs 实例化模块 ui-mask angulajs
- javascript - Ajax 执行更新和删除页面不会自动刷新我正在使用 laravel 5.5
- amazon-s3 - aws cli s3 存储桶删除具有日期条件的对象
- javascript - 使用 nuxtjs 渲染数字
- asynchronous - 龙卷风,使 on_finish() 异步