Spring Boot 整合 JavaFX 核心知识点详解
1. 架构设计与集成模式
1.1 Spring Boot 与 JavaFX 的分层架构设计
Spring Boot 与 JavaFX 的整合需要精心设计的分层架构,以充分利用两个框架的优势。
标准分层架构
┌───────────────────────────────────────────────────┐
│ 表现层 (View Layer) │
│ │
│ ┌─────────────┐ ┌─────────────┐ │
│ │ FXML 文件 │ │ JavaFX 控件 │ │
│ └─────────────┘ └─────────────┘ │
└───────────────────────────────────────────────────┘▲│
┌───────────────────────┼───────────────────────────┐
│ 控制层 (Controller) │
│ │
│ ┌─────────────────────────────────────┐ │
│ │ JavaFX 控制器 (FXML Controller)│ │
│ │ 由Spring管理依赖注入 │ │
│ └─────────────────────────────────────┘ │
└───────────────────────────────────────────────────┘▲│
┌───────────────────────┼───────────────────────────┐
│ 服务层 (Service Layer) │
│ │
│ ┌─────────────┐ ┌─────────────┐ │
│ │Spring Services│ │Spring Components│ │
│ └─────────────┘ └─────────────┘ │
└───────────────────────────────────────────────────┘▲│
┌───────────────────────┼───────────────────────────┐
│ 数据层 (Data Layer) │
│ │
│ ┌─────────────┐ ┌─────────────┐ │
│ │Spring Data │ │Repositories │ │
│ │Repositories │ │ │ │
│ └─────────────┘ └─────────────┘ │
└───────────────────────────────────────────────────┘
职责分离原则
- UI 层(JavaFX):负责用户界面的渲染和用户输入处理
- 控制器层(JavaFX Controller + Spring):处理用户交互,调用服务层
- 服务层(Spring Service):包含业务逻辑,处理事务
- 数据访问层(Spring Data):处理数据持久化
整合策略要点
- UI与业务逻辑分离:JavaFX负责界面,Spring负责业务逻辑
- 控制器作为桥梁:JavaFX控制器由Spring管理,连接UI和服务
- 模型数据共享:使用DTO或专用模型对象在层间传递数据
- 依赖注入贯穿:所有组件使用Spring的依赖注入机制
示例项目结构:
src/main/java
├── com.example.app
│ ├── SpringBootJavaFxApplication.java // 应用入口
│ ├── config
│ │ └── AppConfig.java // Spring配置
│ ├── controller
│ │ └── MainController.java // JavaFX控制器
│ ├── model
│ │ └── User.java // 数据模型
│ ├── repository
│ │ └── UserRepository.java // 数据访问
│ ├── service
│ │ └── UserService.java // 业务逻辑
│ └── view
│ └── StageManager.java // 视图管理
└── resources├── fxml│ └── main.fxml // UI布局└── application.properties // Spring配置
1.2 MVP/MVVM 在整合应用中的实现
Spring Boot 与 JavaFX 的整合特别适合采用MVP或MVVM架构模式。
MVP (Model-View-Presenter) 实现
┌──────────────┐ ┌──────────────┐ ┌──────────────┐
│ Model │◄─────│ Presenter │◄─────│ View │
│ (Spring管理) │─────►│ (Spring管理) │─────►│ (JavaFX UI) │
└──────────────┘ └──────────────┘ └──────────────┘
实现要点:
- Model:由Spring管理的实体和业务逻辑
- View:JavaFX的UI组件,仅负责显示
- Presenter:连接Model和View,处理UI事件和数据转换
代码示例:
// Model (Spring管理)
@Service
public class UserService {private final UserRepository repository;@Autowiredpublic UserService(UserRepository repository) {this.repository = repository;}public List<User> findAllUsers() {return repository.findAll();}
}// Presenter (Spring管理)
@Component
public class UserPresenter {private final UserService userService;private UserView view;@Autowiredpublic UserPresenter(UserService userService) {this.userService = userService;}public void setView(UserView view) {this.view = view;}public void loadUsers() {List<User> users = userService.findAllUsers();view.displayUsers(users);}
}// View (JavaFX)
public class UserViewController implements UserView {@FXMLprivate TableView<User> userTable;@Autowiredprivate UserPresenter presenter;@FXMLpublic void initialize() {presenter.setView(this);}@Overridepublic void displayUsers(List<User> users) {userTable.getItems().setAll(users);}
}
MVVM (Model-View-ViewModel) 实现
┌──────────────┐ ┌──────────────┐ ┌──────────────┐
│ Model │◄─────│ ViewModel │◄─────│ View │
│ (Spring管理) │─────►│ (Spring管理) │─────►│ (JavaFX UI) │
└──────────────┘ └──────────────┘ └──────────────┘▲ ││ ▼┌──────────────┐│ Data Binding │└──────────────┘
实现要点:
- Model:由Spring管理的数据模型和业务逻辑
- View:JavaFX的UI组件
- ViewModel:由Spring管理,包含UI所需的数据和命令
- 绑定:JavaFX的属性绑定机制实现View与ViewModel的双向绑定
代码示例:
// ViewModel (Spring管理)
@Component
public class UserViewModel {private final UserService userService;private final ObservableList<User> users = FXCollections.observableArrayList();private final StringProperty searchText = new SimpleStringProperty("");@Autowiredpublic UserViewModel(UserService userService) {this.userService = userService;// 响应搜索文本变化searchText.addListener((obs, oldValue, newValue) -> {if (newValue != null) {loadFilteredUsers(newValue);}});}public void loadUsers() {List<User> userList = userService.findAllUsers();users.setAll(userList);}private void loadFilteredUsers(String filter) {List<User> filteredList = userService.findByNameContaining(filter);users.setAll(filteredList);}// Getters for observable collections and propertiespublic ObservableList<User> getUsers() {return users;}public StringProperty searchTextProperty() {return searchText;}
}// View (JavaFX)
public class UserView {@FXMLprivate TableView<User> userTable;@FXMLprivate TextField searchField;@Autowiredprivate UserViewModel viewModel;@FXMLpublic void initialize() {// 双向绑定搜索文本searchField.textProperty().bindBidirectional(viewModel.searchTextProperty());// 绑定表格数据userTable.setItems(viewModel.getUsers());// 初始加载数据viewModel.loadUsers();}
}
1.3 整合应用的启动流程设计
Spring Boot 与 JavaFX 的整合涉及到两个框架各自的启动流程,需要精心设计。
启动流程图
┌────────────────────────────────────┐
│ 1. Spring Boot 应用入口点 │
│ public static void main() │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 2. 启动Spring容器 │
│ SpringApplication.run() │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 3. Spring初始化完成 │
│ @PostConstruct │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 4. 启动JavaFX应用 │
│ 启动JavaFX Application │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 5. JavaFX应用初始化 │
│ Application.init() │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 6. JavaFX UI线程启动 │
│ Application.start() │
└─────────────────┬──────────────────┘│▼
┌────────────────────────────────────┐
│ 7. 加载主场景 │
│ 加载FXML并显示主舞台 │
└────────────────────────────────────┘
标准启动代码示例
import javafx.application.Application;
import javafx.application.Platform;
import javafx.stage.Stage;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.context.ConfigurableApplicationContext;public class SpringBootJavaFxApplication extends Application {private ConfigurableApplicationContext applicationContext;@Overridepublic void init() {// 在JavaFX线程初始化之前启动SpringapplicationContext = new SpringApplicationBuilder(MainConfig.class).run();}@Overridepublic void start(Stage primaryStage) {// 从Spring容器中获取StageManagerStageManager stageManager = applicationContext.getBean(StageManager.class);// 设置主舞台并显示主界面stageManager.setPrimaryStage(primaryStage);stageManager.showMainView();}@Overridepublic void stop() {// 关闭Spring容器applicationContext.close();Platform.exit();}// 主入口public static void main(String[] args) {// 启动JavaFX应用launch(args);}
}// Spring Boot配置类
@SpringBootApplication
public class MainConfig {@Beanpublic StageManager stageManager() {return new StageManager();}
}// 舞台管理器
@Component
public class StageManager {@Autowiredprivate ConfigurableApplicationContext springContext;private Stage primaryStage;public void setPrimaryStage(Stage primaryStage) {this.primaryStage = primaryStage;primaryStage.setTitle("Spring Boot + JavaFX应用");}public void showMainView() {try {// 创建FXML加载器FXMLLoader loader = new FXMLLoader(getClass().getResource("/fxml/main.fxml"));// 设置控制器工厂,使用Spring创建控制器实例loader.setControllerFactory(springContext::getBean);Parent root = loader.load();Scene scene = new Scene(root);primaryStage.setScene(scene);primaryStage.show();} catch (IOException e) {e.printStackTrace();}}
}
替代方案:保留Spring Boot的启动方式
@SpringBootApplication
public class MainApplication {private ConfigurableApplicationContext applicationContext;@Beanpublic StageInitializer stageInitializer() {return new StageInitializer();}public static void main(String[] args) {Application.launch(JavaFxApplication.class, args);}
}// JavaFX应用类
public class JavaFxApplication extends Application {private ConfigurableApplicationContext applicationContext;@Overridepublic void init() {applicationContext = SpringApplication.run(MainApplication.class);}@Overridepublic void start(Stage stage) {applicationContext.publishEvent(new StageReadyEvent(stage));}@Overridepublic void stop() {applicationContext.close();Platform.exit();}static class StageReadyEvent extends ApplicationEvent {public StageReadyEvent(Stage stage) {super(stage);}public Stage getStage() {return (Stage) getSource();}}
}// 舞台初始化器
@Component
public class StageInitializer {@Autowiredprivate ConfigurableApplicationContext applicationContext;@EventListenerpublic void onStageReady(StageReadyEvent event) {try {Stage stage = event.getStage();FXMLLoader loader = new FXMLLoader(getClass().getResource("/fxml/main.fxml"));loader.setControllerFactory(applicationContext::getBean);Parent root = loader.load();Scene scene = new Scene(root);stage.setScene(scene);stage.show();} catch (IOException e) {e.printStackTrace();}}
}
2. 依赖注入机制
2.1 Spring 容器与 JavaFX 控制器的桥接
Spring 容器与 JavaFX 控制器的桥接是整合中最关键的环节之一,实现这两个框架的无缝连接。
基本桥接原理
┌─────────────────────┐ ┌─────────────────────┐
│ Spring 容器 │ │ JavaFX 系统 │
│ │ │ │
│ ┌───────────────┐ │ │ ┌───────────────┐ │
│ │ @Component │ │ │ │ FXML │ │
│ │ @Service │ │ │ │ │ │
│ │ @Repository │ │ │ │ │ │
│ └───────────────┘ │ │ └───────┬───────┘ │
│ ▲ │ │ │ │
└──────────┼──────────┘ └──────────┼──────────┘│ ││ ▼│ ┌─────────────────────┐│ │ FXMLLoader ││ │ │└─
相关文章:
Spring Boot 整合 JavaFX 核心知识点详解
1. 架构设计与集成模式 1.1 Spring Boot 与 JavaFX 的分层架构设计 Spring Boot 与 JavaFX 的整合需要精心设计的分层架构,以充分利用两个框架的优势。 标准分层架构 ┌────────────────────────────────────────────────…...
Spring MVC DispatcherServlet 的作用是什么? 它在整个请求处理流程中扮演了什么角色?为什么它是核心?
DispatcherServlet 是 Spring MVC 框架的绝对核心和灵魂。它扮演着前端控制器(Front Controller)的角色,是所有进入 Spring MVC 应用程序的 HTTP 请求的统一入口点和中央调度枢纽。 一、 DispatcherServlet 的核心作用和职责: 请…...
亚马逊英国站FBA费用重构:轻小商品迎红利期,跨境卖家如何抢占先机?
一、政策背景:成本优化成平台与卖家共同诉求 2024年4月,亚马逊英国站(Amazon.co.uk)发布近三年来力度最大的FBA费用调整方案,标志着英国电商市场正式进入精细化成本管理时代。这一决策背后,是多重因素的叠…...
Redis在.NET平台中的各种应用场景
关键点总结 连接管理:所有示例都使用ConnectionMultiplexer来管理Redis连接,它是线程安全的,应该在整个应用程序中重用。 键设计:良好的键命名规范很重要,通常使用冒号分隔的层次结构(如page:home:pv)。 数据序列化&…...
近几年字节测开部分面试题整理
文章目录 前言一、面试问题1. 在浏览器url上写一个地址,请描述一下网络方面有哪些变化2. 堆栈数据存储位置3. HTTP POST请求支持的数据格式4. 缩容要注意些什么?5. Python中元组、数组、list和数组的区别6. Python中is和的区别7. HTTP与HTTPS8. 已知两个…...

艾蒙顿桌面app下载-Emotn UI下载安装-emotn ui官方tv版安卓固件
在智能电视桌面应用的领域里,Emotn UI 凭借其简洁无广告、可自定义等特点,赢得了不少用户的关注。然而,小编深入了解后发现了一款更好用的电视桌面——乐看家桌面在诸多方面更具优势,能为你带来更优质的大屏体验。 乐看家桌面内置…...

3、ArkTS语言介绍
目录 基础知识函数函数声明可选参数Rest参数返回类型箭头函数(又名Lambda函数)闭包 类字段字段初始化getter和setter继承父类访问方法重写方法重载签名可见性修饰符(Public、Private、protected) 基础知识 ArkTS是一种为构建高性…...

修改了Element UI中组件的样式,打包后样式丢失
修改了Element UI中组件的样式,在本地运行没有问题,但是打包到线上发现样式丢失(样式全部不生效、或者有一部分生效,一部分不生效),问题在于css的加载顺序导致代码编译后样式被覆盖了, 解决办法…...
Linux GPIO驱动开发实战:Poll与异步通知双机制详解
1. 引言 在嵌入式Linux开发中,GPIO按键驱动是最基础也最典型的案例之一。本文将基于一个支持poll和异步通知双机制的GPIO驱动框架,深入剖析以下核心内容: GPIO中断与防抖处理环形缓冲区设计Poll机制实现异步通知(SIGIO)实现应用层交互方式 …...

【springsecurity oauth2授权中心】jwt令牌更换成自省令牌 OpaqueToken P4
前言 前面实现了授权中心授权,客户端拿到access_token后就能请求资源服务器接口 权限的校验都是在资源服务器上进行的,授权服务器颁发的access_token有限期是2小时,也就是说在2小时之内,不管授权服务器那边用户的权限如何变更都…...

诱骗协议芯片支持PD2.0/3.0/3.1/PPS协议,支持使用一个Type-C与电脑传输数据和快充取电功能
快充是由充电器端的充电协议和设备端的取电协议进行握手通讯进行协议识别来完成的,当充电器端的充电协议和设备端的取电协议握手成功后,设备会向充电器发送电压请求,充电器会根据设备的需求发送合适的电压给设备快速供电。 设备如何选择快充…...

变量在template里不好使,在setup好使?
问题: 自定义的一个函数 ,import导入后 setup里面使用正常 ,在template里面说未定义 作用域问题 在 Vue 的模板语法中,模板(template )里能直接访问的是组件实例上暴露的属性和方法。从代码看,…...

OpenCV 图形API(53)颜色空间转换-----将 RGB 图像转换为灰度图像函数RGB2Gray()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 将图像从 RGB 色彩空间转换为灰度。 R、G 和 B 通道值的常规范围是 0 到 255。生成的灰度值计算方式如下: dst ( I ) 0.299 ∗ src…...

Trae+DeepSeek学习Python开发MVC框架程序笔记(四):使用sqlite存储查询并验证用户名和密码
继续通过Trae向DeepSeek发问并修改程序,实现程序运行时生成数据库,用户在系统登录页面输入用户名和密码后,控制器通过模型查询用户数据库表来验证用户名和密码,验证通过后显示登录成功页面,验证失败则显示登录失败页面…...

超详细mac上用nvm安装node环境,配置npm
一、安装NVM 打开终端,运行以下命令来安装NVM: curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.5/install.sh | bash 然后就会出现如下代码: > Profile not found. Tried ~/.bashrc, ~/.bash_profile, ~/.zprofile, ~/.…...
STM32 串口通信
引言 在嵌入式系统开发中,串口通信是最基础且重要的通信方式之一。无论是设备调试、模块对接还是远程通信,串口都扮演着关键角色。本文将从通信协议原理出发,结合STM32F4系列MCU,深入讲解串口通信的硬件实现和软件配置࿰…...

hi3516cv610构建音频sample工程代码步骤
hi3516cv610构建音频sample工程代码步骤 sdk版本:Hi3516CV610_SDK_V1.0.1.0 硬件:非es8388 工程代码: 通过网盘分享的文件:audio_easy.zip 链接: https://pan.baidu.com/s/1gx61S_F3-pf6hPyfbGaRXg 提取码: 4gbg --来自百度网盘…...

12.QT-Combo Box|Spin Box|模拟点餐|从文件中加载选项|调整点餐份数(C++)
Combo Box QComboBox 表⽰下拉框 核⼼属性 属性说明currentText当前选中的⽂本currentIndex当前选中的条⽬下标.从0开始计算.如果当前没有条⽬被选中,值为-1editable是否允许修改设为true时, QComboBox 的⾏为就⾮常接近 QLineEdit ,也可以 设置 validatoriconSize下拉框图标…...

UML 顺序图:电子图书馆管理系统的交互之道
目录 一、初识 UML 顺序图 二、电子图书馆管理系统顺序图解析 (一)借阅流程 (二)归还流程 三、顺序图绘画 四、顺序图的优势与价值 五、总结 UML 顺序图是描绘系统组件交互的有力工具。顺序图直观展示消息传递顺序与对象协…...
分布式架构设计与应用:从理论到实践
在云计算、大数据与高并发场景的驱动下,分布式架构已成为现代软件系统的核心技术。它通过将计算、存储与业务逻辑分散到多台机器上,解决了单体架构的扩展性瓶颈与单点故障问题。本文将从设计原则、核心组件到典型应用场景,深入剖析分布式架构…...
Uniapp:view容器(容器布局)
目录 一、基本概述二、属性说明三、常用布局3.1 横向布局3.2 纵向布局3.3 更多布局3.3.1 纵向布局-自动宽度3.3.2 纵向布局-固定宽度3.3.3 横向布局-自动宽度3.3.4 横向布局-居中3.3.5 横向布局-居右3.3.6 横向布局-平均分布3.3.7 横向布局-两端对齐3.3.8 横向布局-自动填充3.3…...

访问者模式:分离数据结构与操作的设计模式
访问者模式:分离数据结构与操作的设计模式 一、模式核心:将操作从数据结构中分离,支持动态添加新操作 在软件开发中,当数据结构(如树、集合)中的元素类型固定,但需要频繁添加新的操作…...

【AI训练环境搭建】在IDE(Pycharm或VSCode)上使用WSL2+Ubuntu22.04+Conda+Tensorflow+GPU进行机器学习训练
本次实践将在IDE(Pycharm或VSCode)上使用WSL2Ubuntu22.04TensorflowGPU进行机器学习训练。基本原理是在IDE中拉起WSL2中的Python解释器,并运行Python程序。要运行CondaTensorflowGPU你可能需要进行以下准备工作。 1. 此示例中将使用一个mnis…...

Leetcode19(亚马逊真题):删除链表的倒是第N个节点
题目分析 删除节点关键:找到被删节点的前一个节点,指针指向 虚拟头节点,方便删除头结点,形成统一操作 为啥要让快指针先行? 我认为更好懂的一种解释:快指针先行n步,这样快慢指针之间形成了一…...

Hadoop+Spark 笔记 2025/4/21
读书笔记 定义 1. 大数据(Big Data) - 指传统数据处理工具难以处理的海量、高速、多样的数据集合,通常具备3V特性(Volume体量大、Velocity速度快、Variety多样性)。扩展后还包括Veracity(真实性&#x…...
千问2.5-VL-7B的推理、微调、部署_笔记2
接上篇:部署千问2.5-VL-7B_笔记1-CSDN博客 这里主要记录微调过程 一、模型微调 这里也使用ms-swift对qwen2.5和qwen2-vl进行自我认知微调和图像OCR微调,并对微调后的模型进行推理。ms-swift是魔搭社区官方提供的LLM工具箱,支持300大语言模…...

Redis从入门到实战基础篇
前言:Redis的安装包含在Redis从入门到实战先导篇中,需要的可移步至此节 目录 1.Redis简单介绍 2.初始Redis 2.1.认识NoSQL 2.2.认识Redis 2.3.安装Redis 3.Redis常见命令 3.1 Redis数据结构 3.2 通用命令 3.3 String命令 3.4 Key的层级结构 3…...
【Docker】在Ubuntu平台上的安装部署
写在前面 docker作为一种部署项目的辅助工具,真是太好用了需要魔法,不然无法正常运行笔者环境:ubuntu22.04 具体步骤 更新系统包索引 sudo apt update安装必要依赖包 sudo apt install -y apt-transport-https ca-certificates curl softwa…...

Java虚拟机(JVM)家族发展史及版本对比
Java虚拟机(JVM)家族发展史及版本对比 一、JVM家族发展史 1. 早期阶段(1996-2000) Classic VM(Java 1.0-1.1): 厂商:Sun Microsystems(Oracle前身)。特点&…...

【学习笔记】Cadence电子设计全流程(三)Capture CIS 原理图绘制(下)
【学习笔记】Cadence电子设计全流程(三)Capture CIS 原理图绘制(下) 3.16 原理图中元件的编辑与更新3.17 原理图元件跳转与查找3.18 原理图常见错误设置于编译检查3.19 低版本原理图文件输出3.20 原理图文件的锁定与解锁3.21 Orca…...