首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 企业软件 > 行业软件 >

java模板方式与策略模式示例

2012-10-09 
java模板模式与策略模式示例模板方法简介:把一些公用的通用的内容抽出来,个性的变动的内容做为参数暴露出

java模板模式与策略模式示例
模板方法简介:
把一些公用的通用的内容抽出来,个性的变动的内容做为参数暴露出来,做为一个模板。在使用时只用传递不同的参数到此模板,便可以得到想要的数据结果,这就是设计模式中得模板方法模式。

下面是使用模板方法来设计dao层的代码示例:

import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;/** * 定义一个抽象的dao父类 *  * @author wb_gaobingyin *  */abstract class BaseDao {/** * 这里把查询对象时的公用的地方抽出来了,做为查询单个对象的一个模板, * 具体的查询将在这个模板上开展,这种处理的方式就叫做模板方法。 * @param sql * @param args * @return */public Object getObject(String sql, Object[] args) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;Object obj = null;try {ps = conn.prepareStatement(sql);rs = ps.executeQuery();for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);// 给sql中得参数赋值}if (rs.next()) {obj = this.RowMapperMethod(rs);// 这个方法调用的时总会调用子类中实现的具体方法,进行不同的字段与属性的映射}} catch (SQLException e) {e.printStackTrace();} finally {// 释放数据库链接对象}return obj;}public abstract Object RowMapperMethod(ResultSet rs) throws SQLException;}class UserDaoImpl extends BaseDao {/** * 根据id查询user对象 *  * @return User */public User getUser() {String sql = "select user_id,user_name,age from uset_tbl where user_id=?";Object[] args = new Object[] { "abing" };User user = (User) this.getObject(sql, args);// 在模板的基础上传入参数,获得想要的结果return user;}/** * 实现基类中的映射方法,不对的实现dao里有不同的装配方式 */public Object RowMapperMethod(ResultSet rs) throws SQLException {User user = new User();user.setUserId(rs.getString("user_id"));user.setUserName(rs.getString("user_name"));user.setAge(rs.getInt("age"));return user;}}class User {private String userId;private String userName;private int age;public String getUserId() {return userId;}public void setUserId(String userId) {this.userId = userId;}public String getUserName() {return userName;}public void setUserName(String userName) {this.userName = userName;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}






上面dao的设计基础上,假如我现在有一个新的需求,即根据userId仅仅查询出userName,那该如何处理,如何拓展,显然上面的模板已经不能够满足了,是不是需要再写一个模板呢。想想抽模板的过程,是把不变的部分抽出来,而变的部分sql,args做为参数暴露出,装配参数也是变的,所以是用抽象方法让子类去实现。目前的问题就出在装配上,能不能把装配的过程也做为一个参数传递呢?这里就需要用的策略模式了。
策略模式简介:
策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。
1、 多个类只区别在表现行为不同,可以使用Strategy模式,在运行时动态选择具体要执行的行为。(例如FlyBehavior和QuackBehavior) 2、 需要在不同情况下使用不同的策略(算法),或者策略还可能在未来用其它方式来实现。(例如FlyBehavior和QuackBehavior的具体实现可任意变化或扩充)  
3、 对客户(Duck)隐藏具体策略(算法)的实现细节,彼此完全独立。

下面是使用策略模式在模板方法之上的dao层改进代码:
import java.sql.Connection;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;/** * 定义一个抽象的dao父类 *  * @author abing *  */class BaseDao {/** * 在模板方法的基础上,为getObject方法添加了一个RowMapper[策略接口]接口属性, * 通过调用方构建不同的RowMapper[策略接口]来完成不同的行为 *  * @param sql * @param args * @return */public Object getObject(String sql, Object[] args, RowMapper rm) {Connection conn = null;PreparedStatement ps = null;ResultSet rs = null;Object obj = null;try {ps = conn.prepareStatement(sql);rs = ps.executeQuery();for (int i = 0; i < args.length; i++) {ps.setObject(i + 1, args[i]);// 给sql中得参数赋值}if (rs.next()) {obj = rm.mapRows(rs);}} catch (SQLException e) {e.printStackTrace();} finally {// 释放数据库链接对象}return obj;}}/** * 定义一个行映射器-[策略接口] 只需要这么一个接口,通过内部类的使用就可以构建不同的方法实现体,来完成不同的事情 *  * @author abing *  */interface RowMapper {public Object mapRows(ResultSet rs) throws SQLException;}class UserDaoImpl extends BaseDao {/** * 根据id查询user对象 *  * @return User */public User getUser() {String sql = "select user_id,user_name,age from uset_tbl where user_id=?";Object[] args = new Object[] { "abing" };// 这里使用匿名内部类User user = (User) this.getObject(sql, args, new RowMapper() {@Overridepublic Object mapRows(ResultSet rs) throws SQLException {// 构建并且返回user对象,这是一种策略User user = new User();user.setUserId(rs.getString("user_id"));user.setUserName(rs.getString("user_name"));user.setAge(rs.getInt("age"));return user;}});return user;}/** * 根据id查询userName属性 *  * @return User */public String getUserName() {String sql = "select user_id,user_name,age from uset_tbl where user_id=?";Object[] args = new Object[] { "abing" };// 构建且仅返回userName字符串对象,这也是一种策略String userName = (String) this.getObject(sql, args, new RowMapper() {@Overridepublic Object mapRows(ResultSet rs) throws SQLException {// TODO Auto-generated method stubreturn rs.getString("user_name");}});return userName;}}class User {private String userId;private String userName;private int age;public String getUserId() {return userId;}public void setUserId(String userId) {this.userId = userId;}public String getUserName() {return userName;}public void setUserName(String userName) {this.userName = userName;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}}

热点排行