news 2026/6/18 17:40:38

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
你写 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 但没跟设计模式对上号,可以搜一下这个小程序。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/18 17:39:41

ERPNext开源ERP系统完全指南:从零开始构建企业数字化平台

ERPNext开源ERP系统完全指南&#xff1a;从零开始构建企业数字化平台 【免费下载链接】erpnext Free and Open Source Enterprise Resource Planning (ERP) 项目地址: https://gitcode.com/GitHub_Trending/er/erpnext 还在为高昂的企业管理软件费用烦恼&#xff1f;ERP…

作者头像 李华
网站建设 2026/6/18 17:35:16

大四学生面试复盘软件2026版零基础使用指南避坑与快速上手

收到同事推荐入手听脑 AI&#xff0c;打开页面功能繁多&#xff0c;不少 HR 新手会不清楚操作顺序。日常高频处理面试记录、OKR 绩效面谈等沟通录音&#xff0c;这份入门实操指南专门面向 HR 从业者&#xff0c;分享一套短时间就能走完录音到完整纪要的标准化流程&#xff0c;同…

作者头像 李华
网站建设 2026/6/18 17:35:01

3步终极指南:GetQzonehistory帮你完整备份QQ空间所有回忆

3步终极指南&#xff1a;GetQzonehistory帮你完整备份QQ空间所有回忆 【免费下载链接】GetQzonehistory 获取QQ空间发布的历史说说 项目地址: https://gitcode.com/GitHub_Trending/ge/GetQzonehistory 你的QQ空间里珍藏着多少青春记忆&#xff1f;那些年发的说说、转发…

作者头像 李华
网站建设 2026/6/18 17:27:48

10分钟永久保存微信聊天记录:留痕工具完全指南

10分钟永久保存微信聊天记录&#xff1a;留痕工具完全指南 【免费下载链接】WeChatMsg 提取微信聊天记录&#xff0c;将其导出成HTML、Word、CSV文档永久保存&#xff0c;对聊天记录进行分析生成年度聊天报告 项目地址: https://gitcode.com/GitHub_Trending/we/WeChatMsg …

作者头像 李华
网站建设 2026/6/18 17:22:48

Maupassant Hugo主题国际化支持:多语言博客搭建完整教程

Maupassant Hugo主题国际化支持&#xff1a;多语言博客搭建完整教程 【免费下载链接】maupassant-hugo Maupassant theme, ported to Hugo. Forked from JokerQyou. 项目地址: https://gitcode.com/gh_mirrors/ma/maupassant-hugo Maupassant Hugo主题是一款优雅的博客主…

作者头像 李华