大多数开发人员现在还在使用if else的过程结构,曾看过jdon的banq大哥写的一篇文章,利用command,aop模式替代if else过程结构。当时还不太明白,这几天看了《重构》第一章的影片租赁案例,感触颇深。下面我来谈一谈为什么要用state pattern替代if else,替代if else有什么好处,以及给出详细代码怎么替代if else。本文参考jdon的“你还在使用if else吗?”及《重构》第一章。
<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" /?>
首先我们模仿影片租赁过程,顾客租凭影片,影片分为儿童片、普通片、新片。根据影片类型及租凭天数价格各不相同(优惠程度不同),用户累计积分不同。
OK
,现在我们使用
if else
表示。
/** */ /**
* <p/> Title:影片基类
* </p>
* <p/> Description:
* </p>
* <p/> Date:2006-10-14 15:47:55
* </p>
*
* @author EOMS 曲静波
* @version 1.0
*/
public class Movie {
// 普通片标识
public static int REGULAR = 1 ;
// 新片标识
public static int NEW_RELEASE = 2 ;
// 儿童片标识
public static int CHILDREN = 3 ;
/** */ /**
* 获取租赁影片总价
*
* @param movieCode
* 影片类型
* @param days
* 租凭天数
* @return 租赁影片总价
* @throws MovieException
* 没有影片类型抛出异常
*/
public double getCharge( int movieCode, int days) throws MovieException {
double result = 0 ;
// 普通片
if (movieCode == Movie.REGULAR)
// 单价为2
{
result = 2 ;
// 如果租赁天数大于2则,则优惠
if (days > 2 ) {
result += (days - 2 ) * 1.5 ;
}
// 返回总价
return result;
}
// 最新发布片
else if (movieCode == Movie.NEW_RELEASE) {
// 新片没有优惠,单价为3
return days * 3 ;
}
// 儿童片
else if (movieCode == Movie.CHILDREN) {
// 影片单价
result = 1.5 ;
// 如果租赁时间大于3天则做价格优惠
if (days > 3 ) {
result += (days - 3 ) * 1.5 ;
}
// 返回租赁影片总价
return result;
} else
throw new MovieException( " 影片不存在 " );
}
/** */ /**
* 获取租赁影片积分
*
* @param movieCode
* 影片类型
* @param days
* 租凭天数
* @return 租赁影片积分
* @throws MovieException
* 没有影片类型抛出异常
*/
public double getIntegral( int movieCode, int days) throws MovieException
{
// 普通片
if (movieCode == Movie.REGULAR)
return days * 2 ;
// 最新发布片
else if (movieCode == Movie.NEW_RELEASE)
return days * 3 ;
// 儿童片
else if (movieCode == Movie.CHILDREN)
return days * 1.5 ;
else
throw new MovieException( " 影片不存在 " );
}
}
OK ,我们看一下,现在的 Movie 完全符合租赁需求,通过 getIntegral(int movieCode,int days) 和 getCharge(int movieCode,int days) 来获得租赁积分及租赁价格。从开闭原则角度来看,如果要添加新的影片类型,我们必须修改 getIntegral(int movieCode,int days) 和 getCharge(int movieCode,int days) 这两个方法。而若要改变租赁价格、积分的优惠规则时,仍需要修改 getIntegral(int movieCode,int days) 和 getCharge(int movieCode,int days) 方法。现在看来,只有三种影片类型,维护还较方便。而当影片类型较多时,例如 10 种, 100 种影片类型,这样就是不可以想像的维护。
现在我们来看一下,使用 state pattern 来代替 if else 。先来个类图。
首先我们建立一个 abstract class Price 做为影片类型的基类,基类中含有两个 abstract 方法,获取总价格 getCharge(int days), 获取总积分 getIntegral(int days) 方法 , 继承 abstract classPrice 的三个影片类型儿童片 class ChilerenPrice, 普通片 class RegularPrice, 最新片 class NewReleasePrice 。分别实现 getCharge(int days),getIntegral(int days) 方法,实现方法写入计算价格的优惠方案及积分的方案。当需要修改方案时,我们只需在某个影片类的方法中对应修改就可以。若新增一个影片分类时,我们只需新增一个实现类实现 abstract class Price 类就 OK 。
class Movie
代表影片,其关联一个
Price
类,而
setPrice(String movieClass)
方法类似于一个工厂类,传入
movieClass
为包名类名,用
java
反射机制实例化一个具体传入
movieClass
的影片类型实现类,这样我们通过这几行代码就可以获得该影片类型的价格和积分。
regularMovie.setPrice(Movie.REGULAR);
System.out.println( " 普通影片租赁10天的价格 " + regularMovie.getPrice().getCharge( 10 ));
System.out.println( " 普通影片租赁10天的积分 " + regularMovie.getPrice().getIntegral( 10 ));
下面我们给出详细代码
abstract class Price价格基类
/** *//**
* <p/> Title:
* </p>
* <p/> Description:
* </p>
* <p/> Date:2006-10-14 15:48:22
* </p>
*
* @author EOMS 曲静波
* @version 1.0
*/
public abstract class Price {
/** *//**
* 获取租赁影片价格需实现该此方法
*
* @param days
* 租赁天数
* @return 返回影片价格
*/
public abstract double getCharge(int days);
/** *//**
* 获取租赁影片积分需实现此方法
*
* @param days
* 租赁天数
* @return 返回影片积分
*/
public abstract double getIntegral(int days);
}
儿童片ChildrenPrice类,实现abstract class Price ,实现儿童片租赁总价getCharge(int days)及儿童片租赁积分getIntegral(int days)。
/** *//**
* <p/> Title:儿童片租赁积分、价格实现
* </p>
* <p/> Description:
* </p>
* <p/> Date:2006-10-14 15:49:04
* </p>
*
* @author EOMS 曲静波
* @version 1.0
*/
public class ChildrenPrice extends Price {
/** *//**
* 儿童片返回租赁积分,儿童片积分规则为: 根据
*/
public double getIntegral(int days) {
// 返回租赁影片积分
return days * 1.5;
}
/** *//**
* 儿童片返回租赁价格
*/
public double getCharge(int days) {
// 影片单价
double result = 1.5;
// 如果租赁时间大于3天则做价格优惠
if (days > 3) {
result += (days - 3) * 1.5;
}
// 返回租赁影片总价
return result;
}
}
普通片RegularlPrice类,实现abstract class Price ,实现普通片租赁总价getCharge(int days)及普通片租赁积分getIntegral(int days)。
/** *//**
* <p/> Title:普通片租赁积分、价格实现
* </p>
* <p/> Description:
* </p>
* <p/> Date:2006-10-14 15:50:10
* </p>
*
* @author EOMS 曲静波
* @version 1.0
*/
public class RegularlPrice extends Price