从模板模式到JdbcTemplate
模板模式初探
关于模板模式,大家可以参阅 模板方法模式深度解析(一)
原始的jdbc
关于原始的jdbc,如下:
import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; public class JDBC_Test2 { public static void main(String[] args) { Connection conn=null; Statement st=null; ResultSet rs=null; String url="jdbc:mysql://localhost:3306/webexample?useUnicode=true&characterEncoding=UTF-8"; String userName="root"; String passWord=""; try { Class.forName("com.mysql.jdbc.Driver"); conn=DriverManager.getConnection(url,userName,passWord); st=conn.createStatement(); rs=st.executeQuery("select * from admininfo"); while (rs.next()) { System.out.println(rs.getString("Aname")); } } catch (ClassNotFoundException e) { e.printStackTrace(); //log4j 具体记录 }catch (SQLException e) { e.printStackTrace(); } finally{ try { if (rs!=null) { rs.close(); rs=null; } if (st!=null) { st.close(); st=null; } if (conn!=null) { conn.close(); conn=null; } } catch (SQLException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } }
我假定朋友们都有一定的编码经验,那么大家再看了最基本的jdbc后,心里应该会有两个想法
1 作为jdbc hello world级别的示例程序,上面的代码写的还是很不错的,该try catch的地方都注意到了
2 如果要大规模使用的话,就比较麻烦了,操作数据库的核心的代码很少,但是核心之前之后的操作却很多。
jdbc连接数据库 获取数据可以分为一下几步:
1 加载驱动 Class.forName("com.mysql.jdbc.driver");
2 获得连接 Connection con=DriverManager.getConnection(url,userName,password);
3 获得Statement Statement st=con.createStatement();
4 执行sql并获得ResultSet rs=st.executeQuery(sql);
5 处理rs中的数据
while (rs.next()) {
System.out.println(rs.getString("Aname"));
}
6 处理异常
7 关闭连接
仔细看看,其实只有第五步是真正核心的,之前之后的都是那种万年不变的代码
使用模板模式
我们用模板模式来改装一下。
package templatemethod1; import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; public abstract class JDBCTemplate { public Object execute(String sql) { String url=""; String userName=""; String password=""; Connection con=null; Statement st=null; ResultSet rs=null; try{ Class.forName("com.mysql.jdbc.driver"); con=DriverManager.getConnection(url,userName,password); st=con.createStatement(); rs=st.executeQuery(sql); //只把核心的第五步 抽象出来 交给子类处理 Object object=doResultSet(rs); return object; } catch (ClassNotFoundException e){ e.printStackTrace(); } catch(SQLException e){ e.printStackTrace(); } finally{ try{ if(rs!=null){ rs.close(); rs=null; } //... }catch(SQLException e){ e.printStackTrace(); } } return null; } //抽象出来 交给子类处理 public abstract Object doResultSet(ResultSet rs); }
package templatemethod1; import java.sql.ResultSet; import java.sql.SQLException; import java.util.ArrayList; import java.util.List; import model.Student; public class JDBCTemplateStudentImpl extends JDBCTemplate { //doResultSet 就是关于获取Student的真正核心的代码 @Override public Object doResultSet(ResultSet rs) { List<Student> userList = new ArrayList<Student>(); try { Student student = null; while (rs.next()) { student = new Student(); student.setId(rs.getInt("id")); student.setBirth(rs.getDate("birth")); userList.add(student); } return userList; } catch (SQLException e) { e.printStackTrace(); return null; } } }
最后的测试代码
package templatemethod1; import java.util.List; import model.Student; public class TemplateMethodTest { public static void main(String[] args) { String sql="select * from user"; JDBCTemplate template=new JDBCTemplateStudentImpl(); List<Student> students=(List<Student>) template.execute(sql); } }
model.Student的代码我就不给出了,就是几个字段,getset方法而已。
回调
上面的模板方法OK不?
好着呢。
问题是,如果JDBCTemplate中有很多个抽象方法,那么我们就得重新很多方法。太累。
那我们能否只覆盖我们想要的呢?
这里就牵扯到一个名词,callback(回调)
所谓回调,就是方法参数中传递一个接口,父类在调用此方法时,必须调用方法中传递的接口的实现类。
我们直接看代码
package templatemethod2; import java.sql.SQLException; import java.sql.Statement; public interface StatementCallback { //这个doInStatement相当于上面的doResultSet Object doInStatement(Statement stmt) throws SQLException; }
package templatemethod2; import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; public class JDBCTemplate { public Object execute(StatementCallback action) { String url=""; String userName=""; String password=""; Connection con=null; Statement st=null; ResultSet rs=null; try{ Class.forName("com.mysql.jdbc.driver"); con=DriverManager.getConnection(url,userName,password); st=con.createStatement(); Object object=action.doInStatement(st); return object; } //省略try catch return null; } public Object query(StatementCallback action){ return execute(action); } } 下面的测试方法,只要给定一个sql,就能传回List<Student> @SuppressWarnings("unchecked") public List<Student> test2(final String sql) { JDBCTemplate jdbcTemplate = new JDBCTemplate(); return (List<Student>) jdbcTemplate.execute(new StatementCallback() { @Override public Object doInStatement(Statement stmt) throws SQLException { ResultSet rs = stmt.executeQuery(sql); List<Student> userList = new ArrayList<Student>(); Student user = null; while (rs.next()) { user = new Student(); user.setId(rs.getInt("id")); user.setBirth(rs.getDate("birth")); userList.add(user); } return userList; } }); }
参考资料
http://blog.csdn.net/lovelion/article/details/8299794
http://www.bubuko.com/infodetail-666886.html
从模板模式到JdbcTemplate的更多相关文章
- 模板模式(Template)
行为型:Template(模板模式) 作为一个曾经爱好写文章,但是不太懂得写文章的人,我必须承认,开头是个比较难的起步. 模板模式常规定义:模板模式定义了一个算法步骤,把实现延迟到子类. 事实上模板模 ...
- java设计模式之模板模式以及钩子方法使用
1.使用背景 模板方法模式是通过把不变行为搬到超类,去除子类里面的重复代码提现它的优势,它提供了一个很好的代码复用平台.当不可变和可变的方法在子类中混合在一起的时候, 不变的方法就会在子类中多次出现, ...
- Spring中常见的设计模式——模板模式
一.模板模式的应用场景 模板模式又叫模板方法模式(Template Method Pattern),指定义一个算法的骨架,并允许自雷为一个或者多个步骤提供实现.模板模式使得子类可以在不改变算法结果的情 ...
- Spring中的设计模式:模板模式
导读 模板模式在是Spring底层被广泛的应用,比如事务管理器的实现,JDBC模板的实现. 文章首发于作者的微信公众号[码猿技术专栏] 今天就来谈谈「什么是模板模式」.「模板模式的优缺点」.「模板模式 ...
- JAVA设计模式之模板模式
在阎宏博士的<JAVA与模式>一书中开头是这样描述模板方法(Template Method)模式的: 模板方法模式是类的行为模式.准备一个抽象类,将部分逻辑以具体方法以及具体构造函数的形式 ...
- Java设计模式之模板模式(Template )
前言: 最近学习了Glide开源图片缓存框架,在学习到通过使用ModelLoader自定义数据源的时候,Glide巧妙的使用了Java的模板模式来对外暴露处理不同的Url数据源,今天来学习总结一下模板 ...
- Java设计模式(七) 模板模式
[模板模式]在一个方法中定义了一个算法的骨架,而将一些步骤延迟到子类中.模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤. 1,定义模板类 package com.pattern ...
- 模板模式与策略模式/template模式与strategy模式/行为型模式
模板模式 模版模式,又被称为模版方法模式,它可以将工作流程进行封装,并且对外提供了个性化的控制,但主流程外界不能修改,也就是说,模版方法模式中,将工作的主体架构规定好,具体类可以根据自己的需要,各自去 ...
- 12. 星际争霸之php设计模式--模板模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
随机推荐
- PHP date() 函数
实例 格式化本地日期和时间,并返回格式化的日期字符串: <?php // Prints the dayecho date("l") . "<br>&qu ...
- JMeter(十三)-代理服务器录制脚本
今天重点说一下jmeter如何利用自身的代理服务器录制脚本 1:工作台下创建代理服务器 2:配置代理,选择录制控制器 3:在Requests FIltering下添加排除模式,配置正则表达式.否则会录 ...
- Git幕后的“故事”
因为做操作系统实验的原因,所以通读了一遍<Understanding git conceptually>,觉得确实不错,于是就简单地记录一下.有的地方理解的还不是很深,可能不够准确,等抽时 ...
- 初识mybatis(二)
上篇博客我们介绍通过Java代码来创建mybatis的配置文件,港真,这种方式看起来有意思实际在开发中用的并不多,mybatis的配置还是以xml配置为主,本文我们就来看看如何通过xml文件来配置my ...
- 安卓高级3 Android应用Design Support Library完全使用实例
原作者:http://www.open-open.com/lib/view/open1433385856119.html 1 背景 上周一年一度的Google IO全球开发者大会刚刚结束,Google ...
- 一起聊聊什么是P问题、NP问题、NPC问题
概念 P问题:如果一个问题可以找到一个能在多项式的时间里解决它的算法,那么这个问题就属于P问题.通常NOI和NOIP不属于P类问题,我们常见到的一些信息奥赛的题目都是P问题. NP问题:可以在多项式的 ...
- 深度学习&机器学习资源汇总1
本篇博客的目地,是对工作学习过程中所遇所见的一些有关深度学习.机器学习的优质资源,作分类汇总,方便自己查阅,也方便他人学习借用. 主要会涉及一些优质的理论书籍和论文.一些实惠好用的工具库和开源库.一些 ...
- Hexo写博客
hexo配置github Git Install hexo-deployer-git. $ npm install hexo-deployer-git –save 配置_config.yml文件 de ...
- UE4实现闪烁效果
官网文档链接:http://docs.unrealengine.com/latest/CHN/Engine/Rendering/Materials/ExpressionReference/Math/i ...
- Unity UGUI图文混排源码(四) -- 聊天气泡
这里有同学建议在做聊天气泡时,可以更改为一张图集对应多个Text,这样能节省资源,不过我突然想到每个Text一个图集,可以随时更换图集,这样表情图更丰富一些,于是我就先将现有的聊天demo改为了聊天气 ...