大多数开发人员现在还在使用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 表示。
package
com.qujingbo.movie;


/** */
/**
* <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 的影片类型实现类,这样我们通过这几行代码就可以获得该影片类型的价格和积分。
Movie regularMovie
=
new
Movie();
regularMovie.setPrice(Movie.REGULAR);
System.out.println(
"
普通影片租赁10天的价格
"
+
regularMovie.getPrice().getCharge(
10
));
System.out.println(
"
普通影片租赁10天的积分
"
+
regularMovie.getPrice().getIntegral(
10
));
下面我们给出详细代码
abstract class Price价格基类
package
com.qujingbo.movie;


/** */
/**
* <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)
。
package
com.qujingbo.movie;


/** */
/**
* <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)。
package
com.qujingbo.movie;


/** */
/**
* <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