当前位置: 首页 > news >正文

你写 JdbcTemplate 的 callback 写了三年——这就是模板方法,但你从没把它当设计模式

# 你写 JdbcTemplate 的 callback 写了三年——这就是模板方法,但你从没把它当设计模式 大多数 Java 程序员对模板方法模式的认知停留在"定义一个抽象方法让子类实现"。这个认知没错,但只覆盖了模板方法 20% 的用法——剩下的 80% 藏在各种 Spring 组件里,你天天在用却从来没把它跟设计模式挂钩。 ## Spring JdbcTemplate:模板方法的集大成者 写一段你最熟悉的代码: ```java jdbcTemplate.query("SELECT * FROM orders WHERE status = ?", ps -> ps.setString(1, "PAID"), // 步骤1:设置参数 (rs, rowNum) -> { // 步骤2:处理每一行结果 Order order = new Order(); order.setId(rs.getLong("id")); order.setAmount(rs.getBigDecimal("amount")); return order; } ); ``` 这段代码里,`query()` 方法里面发生了什么? 1. 获取连接 2. 创建 PreparedStatement 3. 设置参数(你传进去的 lambda) 4. 执行查询 5. 遍历 ResultSet,每行调你的 RowMapper 6. 关闭 ResultSet 7. 关闭 Statement 8. 归还连接到连接池 总共 8 个步骤,你只关心第 3 步和第 5 步。剩下的 6 个步骤是 JdbcTemplate 帮你写好的——这就是模板方法模式的本质:**固定流程 + 可变步骤**。 把 `JdbcTemplate.query()` 简化后是这样的: ```java public List query(String sql, PreparedStatementSetter pss, RowMapper rowMapper) { Connection conn = null; PreparedStatement ps = null; ResultSet rs = null; try { // 固定步骤1-2:获取连接、创建 Statement conn = dataSource.getConnection(); ps = conn.prepareStatement(sql); // 可变步骤3:设置参数——你来决定 pss.setValues(ps); // 固定步骤4:执行查询 rs = ps.executeQuery(); List results = new ArrayList<>(); int rowNum = 0; while (rs.next()) { // 可变步骤5:映射每一行——你来决定 results.add(rowMapper.mapRow(rs, rowNum++)); } return results; } catch (SQLException e) { throw new DataAccessException(e); } finally { // 固定步骤6-8:关闭资源 closeQuietly(rs); closeQuietly(ps); closeQuietly(conn); } } ``` 这就是模板方法的 callback 变体。GoF 原版是用继承实现——父类定义模板方法,子类重写抽象方法。Spring 升级成了组合——把可变步骤抽成 callback 接口,通过参数传进来。好处是你不用为了每种查询都写一个子类,直接传 lambda。 ## ApplicationContext 的 refresh() 是 Spring 的启动模板 Spring 容器的启动流程是模板方法模式的典型应用: ```java // AbstractApplicationContext.refresh() —— 模板方法 public void refresh() { prepareRefresh(); // 1. 准备刷新 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // 2. 获取 BeanFactory prepareBeanFactory(beanFactory); // 3. 准备 BeanFactory postProcessBeanFactory(beanFactory); // 4. 后处理(模板方法——子类可重写) invokeBeanFactoryPostProcessors(beanFactory); // 5. 执行 BeanFactoryPostProcessor registerBeanPostProcessors(beanFactory); // 6. 注册 BeanPostProcessor initMessageSource(); // 7. 初始化消息源 initApplicationEventMulticaster(); // 8. 初始化事件广播器 onRefresh(); // 9. 模板方法——留给子类的钩子 registerListeners(); // 10. 注册监听器 finishBeanFactoryInitialization(beanFactory); // 11. 实例化所有单例 Bean finishRefresh(); // 12. 完成刷新 } ``` 注意第 4 步和第 9 步——`postProcessBeanFactory()` 和 `onRefresh()` 都是空的 protected 方法,专门留给子类扩展。`AbstractRefreshableWebApplicationContext` 在 `postProcessBeanFactory()` 里注册了 request/session scope,`SpringApplication` 的嵌入式 Web 容器在 `onRefresh()` 里启动了 Tomcat。 框架负责固定流程(启动顺序、异常处理、生命周期管理),你只关心可变部分(注册额外 scope、启动内嵌容器)。这就是模板方法的威力——不是因为算法多复杂,而是因为它把一个容易写乱的流程用固定的框架组织起来了。 ## 但模板方法的继承版本有两个致命问题 GoF 原版的模板方法是基于继承的: ```java public abstract class DataExporter { // 模板方法——final 防止子类改流程 public final void export() { List data = fetchData(); String formatted = formatData(data); // 可变步骤1 validate(formatted); // 可变步骤2 write(formatted); // 可变步骤3 } protected abstract List fetchData(); protected abstract String formatData(List data); protected abstract void validate(String content); protected abstract void write(String content); } public class ExcelExporter extends DataExporter { // 实现四个抽象方法 } public class PdfExporter extends DataExporter { // 实现四个抽象方法 } ``` 问题一:**一个子类只能重写一个模板方法的行为。** 如果 Excel 导出有"带有表头"和"不带表头"两种变体,你怎么办?再写两个 ExcelExporter 的子类?这就是类爆炸。 问题二:**所有子类必须实现所有抽象方法。** 哪怕你的 PdfExporter 不需要 `validate()` 这一步,你也得写个空方法。Java 8 的 default 方法可以缓解,但模板方法的核心痛点是:当你需要**组合**可变行为而非**继承**时,继承是错的。 这就是为什么 Spring 用了 callback 版本——把可变步骤抽成接口,用组合替代继承: ```java public class DataExporter { public void export(DataFetcher fetcher, DataFormatter formatter, DataWriter writer) { List data = fetcher.fetch(); String formatted = formatter.format(data); writer.write(formatted); } } // 使用时 exporter.export( () -> jdbcTemplate.query("SELECT ...", rowMapper), // fetch data -> jsonMapper.writeValueAsString(data), // format content -> Files.write(path, content.getBytes()) // write ); ``` 模板方法变成了策略模式的变体。但核心思路没变:**固定骨架不动,可变细节外挂。** ## 什么时候用模板方法、什么时候用策略模式 这两个经常被搞混,因为看起来都是在"替换算法"。区别在这: - **模板方法**:流程固定,步骤可变。你控制不了顺序(比如必须先 open 再 execute 最后 close),只能替换某一步的行为。 - **策略模式**:整个算法可以整体替换。你可以选冒泡排序也可以选快排,顺序是你自己决定的。 判断标准很简单:看调用者是不是你自己写的。JdbcTemplate 的模板是你写的还是 Spring 写的?Spring 写的,你只填空。策略模式里的排序算法是你选的,你调用的——步骤和顺序都是你掌控的。 我在做一个用卡皮巴拉讲设计模式的小程序「爪爪代码冒险记」,模板方法这章用"做菜"来讲——固定步骤是洗菜→切菜→炒→装盘,但"放什么调料"是你决定的。如果你经常在 Spring 源码里看到各种 callback 但没跟设计模式对上号,可以搜一下这个小程序。
http://www.gsyq.cn/news/1549110.html

相关文章:

  • 海口旧金金条回收攻略,持证实体安全交易 - 开心测评
  • 从创意到原型:如何用Pencil Project高效设计用户界面
  • 指纹浏览器 vs 云手机:核心区别、优缺点及场景选择指南
  • 2026成都本地名表回收保值梯队划分,你的表属于第几档? - 逸程
  • Django毕业设计-基于 Django+Vue 的智慧农业管理系统的设计与实现 基于 Django+Vue 的现代化农业管理平台的设计与实现(源码+LW+部署文档+全bao+远程调试+代码讲解等)
  • Path of Building PoE2:流放之路2角色构建的终极规划工具
  • 黄金变现必看!上海本地人都去的黄金回收门店-收的顶行业标杆持证鉴定 - 奢侈品回收评测
  • Drupal核心SQL注入漏洞CVE-2026-9082深度剖析与防御实战
  • 2026年苏州公司注销代办挑选指南:值得关注的效率与合规双优企业 - 资讯速览
  • Maupassant Hugo主题多评论系统集成:Disqus、utteranc、Waline全面对比
  • 电子工程师如何高效利用数据手册与厂商生态进行硬件开发
  • 5分钟掌握暗黑2存档编辑器:新手可视化修改完整教程
  • 食物图像分类代码实战
  • 2026年祁县家装公司排名:口碑好、设计强、施工扎实的都在这里 - 资讯报道
  • 解锁音乐社交:YouTube Music for Desktop 的 Discord RPC 集成教程
  • ModelScope命令行工具:5个实用技巧快速掌握AI模型管理
  • YTPro的贡献指南:提交PR的步骤与代码审查流程
  • jQuery Anystretch:终极响应式背景图片插件完全指南
  • 2026年 江苏锯条/高铁配件/纺织配件厂家推荐榜:碳钢锯条,合金锯条源头厂实力与品质深度解析 - 品牌发掘
  • Node.js Dialogflow API完全指南:从入门到精通的终极教程
  • PTAM-GPL地图构建(MapMaker)详解:如何实现高效的关键帧管理
  • [智能体-452]:Coze 记忆单元 + 知识库单元:降幻觉、省 Token 底层原理详解
  • Edge-Monitor源码解析:Windows API调用与进程管理技术的实现细节
  • 2026年镇江黄金回收榜单:全城口碑商户综合实力排名 - 生活测评君
  • 24VL014 EEPROM在低功耗物联网节点中的选型、驱动与避坑指南
  • 吸水棒选购指南:如何挑选优质吸水棒 - 热点速览
  • 30分钟快速1:1 复刻企业级 DevOps 架构实战(二)启动devops各组件平台
  • Convoviz可视化功能详解:创建词云和使用图表分析对话模式
  • 如何高效获取官方macOS安装文件:跨平台下载工具完全指南
  • 东莞新手出售翡翠攻略,2026连锁回收实体店资质齐全交易隐私安全 - 名奢变现站