读《研磨设计模式》-代码笔记-模板方法模式

[b]声明:
本文只为方便我个人查阅和理解,详细的分析以及源代码请移步 原作者的博客[url]http://chjavach.iteye.com/[/url][/b]




import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;

/**
* 模板方法模式:
* 采用继承的方式(Template类是一个抽象类)实现这一点:将逻辑(算法)框架放在抽象基类中,并定义好细节的接口,子类中实现细节
* 与工厂方法模式非常相似。只不过模板方法模式关注算法,而工厂方法关注对象的创建(factoryMethod()里面是返回一个产品对象)
* 与策略模式相比,策略模式的策略是平行、平等的,而且一个策略是一个完整的算法,相互之间可切换;
* 但模板方法模式的算法框架是固定的,只是算法的部分细节不同
*
* 以下代码考虑这样一个需求:
* 验证用户登录,分为普通用户和特殊用户,特殊用户的密码是加密的
*/

//============1.=================
//如果需要更多验证(例如页面上的验证码),简单起见,可分别extends LoginModel和LoginTemplate
class LoginModel {

private String userID;
private String password;

public String getUserID() {
return userID;
}

public void setUserID(String userID) {
this.userID = userID;
}

public String getPassword() {
return password;
}

public void setPassword(String password) {
this.password = password;
}
}


abstract class LoginTemplate {

public abstract LoginModel findLoginModel(String userID);

//普通用户登陆(子类)时,密码不加密,直接返回密码就OK。特殊用户则覆盖此方法加密密码
public String encryptPassword(String password) {
return password;
}

public final boolean loginVerify(LoginModel lm) {
boolean pass = false;
LoginModel dblm = this.findLoginModel(lm.getUserID());
if (dblm != null) {
String pwd = encryptPassword(lm.getPassword());
lm.setPassword(pwd);
pass = this.match(lm, dblm);
}
return pass;
}

public boolean match(LoginModel lm, LoginModel dblm) {
return lm.getUserID().equals(dblm.getUserID())
&& lm.getPassword().equals(dblm.getPassword());

//...可有更多验证
}
}


class NormalLoginTemplate extends LoginTemplate {

public LoginModel findLoginModel(String userID) {

//模拟从数据库取数据
LoginModel dblm = new LoginModel();
dblm.setUserID(userID);
dblm.setPassword("passwordInDB");
return dblm;
}

//encryptPassword方法就不需要重写
}


class SpecialLoginTemplate extends LoginTemplate {

public LoginModel findLoginModel(String userID) {

//模拟从数据库取数据
LoginModel dblm = new LoginModel();
dblm.setUserID(userID);
dblm.setPassword("passwordInDB");
return dblm;
}

//重写方法。密码加密。match()方法是比较加密后的密码是否一致
public String encryptPassword(String pwd) {
System.out.println("encrypting password...");
return pwd; //实际应用中返回加密后的密码。这里简单地返回原密码
}

}


//============2.=================
/*
* 用接口回调的方法来实现
*
* 这下面代码看着有些别扭:LoginCallBack里面转调Template的方法,但Template里面又转调LoginCallBack的方法
* 但正是这样,通过LoginCallBack的不同实现,达到了模板模式“子类实现细节”的目的
*/
interface LoginCallBack {

LoginModel findLoginModel(String userID);

//注意传递了一个LoginTemplate2
String encryptPassword(String pwd, LoginTemplate2 template);

boolean match(LoginModel lm, LoginModel lmdb, LoginTemplate2 template);

}


class LoginTemplate2 {

//传入一个LoginCallBack,实际调用时以内部类的形式实现接口里的方法
public final boolean verifyLogin(LoginModel lm, LoginCallBack callback) {
LoginModel lmdb = callback.findLoginModel(lm.getUserID());
String password = callback.encryptPassword(lm.getPassword(), this);
lmdb.setPassword(password);
return callback.match(lm, lmdb, this);
}

public boolean match(LoginModel lm, LoginModel dblm) {
return lm.getUserID().equals(dblm.getUserID())
&& lm.getPassword().equals(dblm.getPassword());
}

//可根据实际情况看是否要重写
public String encryptPassword(String pwd) {
return pwd;
}
}


//客户端。测试
public class TemplateMethodPattern {

public static void main(String[] args) {

//测试情况1
//制造测试数据-一般用户登录
LoginModel userA = new LoginModel();
userA.setUserID("user");
userA.setPassword("passwordInDB");

LoginTemplate template = new NormalLoginTemplate();
boolean pass = template.loginVerify(userA);

System.out.println(userA.getUserID() + " login success?" + pass);

//制造测试数据-特殊用户登录
LoginModel userB = new LoginModel();
userB.setUserID("admin");
userB.setPassword("passwordInDB");

LoginTemplate sTemplate = new SpecialLoginTemplate();
pass = sTemplate.loginVerify(userB);
System.out.println(userB.getUserID() + " login success?" + pass);


//测试情况2-接口回调实现
LoginTemplate2 template2 = new LoginTemplate2();

boolean pass2 = template2.verifyLogin(userA, new LoginCallBack(){

public String encryptPassword(String pwd, LoginTemplate2 template) {
return template.encryptPassword(pwd); //自己不需要实现,转调template里面的默认实现
}

public LoginModel findLoginModel(String userID) {
//从数据库取LoginModel,简单示意一下
LoginModel userAA =new LoginModel();
userAA.setUserID(userID);
userAA.setPassword("passwordInDB");
return userAA;
}

public boolean match(LoginModel lm, LoginModel lmdb,
LoginTemplate2 template) {
return template.match(lm, lmdb);
}

});
System.out.println(userA.getUserID() + " login success?" + pass2);

pass2 = template2.verifyLogin(userB, new LoginCallBack(){

//特殊用户登陆,重写加密密码
public String encryptPassword(String pwd, LoginTemplate2 template) {
System.out.println("encrypting password...");
String encryptedPwd = pwd; //模拟加密
return encryptedPwd;
}

public LoginModel findLoginModel(String userID) {
//从数据库取LoginModel,简单示意一下
LoginModel userBB =new LoginModel();
userBB.setUserID(userID);
userBB.setPassword("passwordInDB");
return userBB;
}

public boolean match(LoginModel lm, LoginModel lmdb,
LoginTemplate2 template) {
return template.match(lm, lmdb);
}

});
System.out.println(userB.getUserID() + " login success?" + pass);

//书上认为Collections.sort是模板模式的一种实现,我觉得这一点不好理解
//stackoverflow上有帖子认为:All non-abstract methods of java.util.AbstractList是模板模式
List<LoginModel> list = new ArrayList<LoginModel>();
list.add(userA);
list.add(userB);
Collections.sort(list, new Comparator<LoginModel>(){
public int compare(LoginModel arg0, LoginModel arg1) {
return 0;
}
});
}

}


/*
很早就知道jdbc用到template模式了,但之前还不是很清楚
其实就是把公用的操作(获取数据库连接,执行sql语句,为sql语句参数赋值等等)写到抽象的Template去,
具体的sql以及sql的参数由子类覆写提供

书上的例子有点复杂,我参照网上写了一个简单的示意:
这个JDBCTemplate只实现查找功能,其他操作略去(增、删、改、数据库连接的释放等)
*/
abstract class DaoTemplate {

/**
* 根据条件查询(如果指定条件)
* @param sql
* @param args 条件参数
* @return
*/
public Object findObject(String sql, Object[] args){
Object obj = null;
Connection conn = null;
PreparedStatement ps = null;
ResultSet rs = null;
try {

//简单模拟获取Connection操作,更多情况下是写一个JdbcUtil
conn = DriverManager.getConnection("url", "user", "pwd");
ps = conn.prepareStatement(sql);

//设置查询参数(条件)
for (int i = 0, size = args.length; i < size; i++) {
ps.setObject(i + 1, args[i]);
}

rs = ps.executeQuery();

if (rs.next()) {
obj = rowMapper(rs); //对结果集进行处理,返回实际的业务对象
}
} catch (SQLException e) {
e.printStackTrace();
}
return obj;
}

abstract protected Object rowMapper(ResultSet rs);
}


class LoginModelDaoImpl extends DaoTemplate {

public LoginModel findLoginModel(String userID) throws SQLException {
String sql = "select userid,password from loginmodel where userid=?";
Object[] args = new String[]{userID};
LoginModel lm =(LoginModel) super.findObject(sql, args);
return lm;
}

@Override
protected Object rowMapper(ResultSet rs){
LoginModel lm = new LoginModel();
try {
lm.setUserID(rs.getString("userid"));
lm.setPassword(rs.getString("password"));
} catch (SQLException e) {
e.printStackTrace();
}
return lm;
}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值