`
seaizon
  • 浏览: 145197 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

研磨设计模式之装饰模式-1(转)

阅读更多

装饰模式(Decorator)

1  场景问题

1.1  复杂的奖金计算

        考虑这样一个实际应用:就是如何实现灵活的奖金计算。
        奖金计算是相对复杂的功能,尤其是对于业务部门的奖金计算方式,是非常复杂的,除了业务功能复杂外,另外一个麻烦之处是计算方式还经常需要变动,因为业务部门经常通过调整奖金的计算方式来激励士气。
        先从业务上看看现有的奖金计算方式的复杂性:

  • 首先是奖金分类:对于个人,大致有个人当月业务奖金、个人累计奖金、个人业务增长奖金、及时回款奖金、限时成交加码奖金等等;
  • 对于业务主管或者是业务经理,除了个人奖金外,还有:团队累计奖金、团队业务增长奖金、团队盈利奖金等等。
  • 其次是计算奖金的金额,又有这么几个基数:销售额、销售毛利、实际回款、业务成本、奖金基数等等;
  • 另外一个就是计算的公式,针对不同的人、不同的奖金类别、不同的计算奖金的金额,计算的公式是不同的,就算是同一个公式,里面计算的比例参数也有可能是不同的。

 

1.2  简化后的奖金计算体系 

        看了上面奖金计算的问题,所幸我们只是来学习设计模式,并不是真的要去实现整个奖金计算体系的业务,因此也没有必要把所有的计算业务都罗列在这里,为了后面演示的需要,简化一下,演示用的奖金计算体系如下:

  • 每个人当月业务奖金 = 当月销售额 X  3%
  • 每个人累计奖金 = 总的回款额 X  0.1%
  • 团队奖金 = 团队总销售额 X 1%

 

1.3  不用模式的解决方案

        一个人的奖金分成很多个部分,要实现奖金计算,主要就是要按照各个奖金计算的规则,把这个人可以获取的每部分奖金计算出来,然后计算一个总和,这就是这个人可以得到的奖金。
(1)为了演示,先准备点测试数据,在内存中模拟数据库,示例代码如下:

Java代码
  1. /**  
  2.  * 在内存中模拟数据库,准备点测试数据,好计算奖金  
  3.  */   
  4. public   class  TempDB {  
  5.     private  TempDB(){  
  6. }  
  7.     /**  
  8.      * 记录每个人的月度销售额,只用了人员,月份没有用  
  9.      */   
  10.     public   static  Map<String,Double> mapMonthSaleMoney =   
  11. new  HashMap<String,Double>();  
  12.     static {  
  13.         //填充测试数据   
  14.         mapMonthSaleMoney.put("张三" , 10000.0 );  
  15.         mapMonthSaleMoney.put("李四" , 20000.0 );  
  16.         mapMonthSaleMoney.put("王五" , 30000.0 );  
  17.     }  
  18. }  
/**
 * 在内存中模拟数据库,准备点测试数据,好计算奖金
 */
public class TempDB {
	private TempDB(){
}
	/**
	 * 记录每个人的月度销售额,只用了人员,月份没有用
	 */
	public static Map<String,Double> mapMonthSaleMoney = 
new HashMap<String,Double>();
	static{
		//填充测试数据
		mapMonthSaleMoney.put("张三",10000.0);
		mapMonthSaleMoney.put("李四",20000.0);
		mapMonthSaleMoney.put("王五",30000.0);
	}
}

 

(2)按照奖金计算的规则,实现奖金计算,示例代码如下:

Java代码
  1. /**  
  2.  * 计算奖金的对象  
  3.  */   
  4. public   class  Prize {  
  5.     /**  
  6.      * 计算某人在某段时间内的奖金,有些参数在演示中并不会使用,  
  7.      * 但是在实际业务实现上是会用的,为了表示这是个具体的业务方法,  
  8.      * 因此这些参数被保留了  
  9.      * @param user 被计算奖金的人员  
  10.      * @param begin 计算奖金的开始时间  
  11.      * @param end 计算奖金的结束时间  
  12.      * @return 某人在某段时间内的奖金  
  13.      */   
  14.     public    double  calcPrize(String user,Date begin,Date end){  
  15.         double  prize =  0.0 ;   
  16.         //计算当月业务奖金,所有人都会计算   
  17.         prize = this .monthPrize(user, begin, end);  
  18.         //计算累计奖金   
  19.         prize += this .sumPrize(user, begin, end);  
  20.           
  21.         //需要判断该人员是普通人员还是业务经理,团队奖金只有业务经理才有   
  22.         if ( this .isManager(user)){  
  23.             prize += this .groupPrize(user, begin, end);  
  24.         }  
  25.         return  prize;  
  26.     }  
  27.   
  28.     /**  
  29.      * 计算某人的当月业务奖金,参数重复,就不再注释了  
  30.      */   
  31.     private   double  monthPrize(String user, Date begin, Date end) {  
  32.         //计算当月业务奖金,按照人员去获取当月的业务额,然后再乘以3%   
  33.         double  prize = TempDB.mapMonthSaleMoney.get(user) *  0.03 ;  
  34.         System.out.println(user+"当月业务奖金" +prize);  
  35.         return  prize;  
  36.     }  
  37.   
  38.     /**  
  39.      * 计算某人的累计奖金,参数重复,就不再注释了  
  40.      */   
  41.     public   double  sumPrize(String user, Date begin, Date end) {  
  42.         //计算累计奖金,其实应该按照人员去获取累计的业务额,然后再乘以0.1%   
  43.         //简单演示一下,假定大家的累计业务额都是1000000元   
  44.         double  prize =  1000000  *  0.001 ;  
  45.         System.out.println(user+"累计奖金" +prize);  
  46.         return  prize;  
  47.     }     
  48.   
  49.     /**  
  50.      * 判断人员是普通人员还是业务经理  
  51.      * @param user 被判断的人员  
  52.      * @return true表示是业务经理,false表示是普通人员  
  53.      */   
  54.     private   boolean  isManager(String user){  
  55.         //应该从数据库中获取人员对应的职务   
  56.         //为了演示,简单点判断,只有王五是经理   
  57.         if ( "王五" .equals(user)){  
  58.             return   true ;              
  59.         }  
  60.         return   false ;  
  61.     }  
  62.     /**  
  63.      * 计算当月团队业务奖,参数重复,就不再注释了  
  64.      */   
  65.     public   double  groupPrize(String user, Date begin, Date end) {  
  66.         //计算当月团队业务奖金,先计算出团队总的业务额,然后再乘以1%,   
  67. //假设都是一个团队的   
  68.         double  group =  0.0 ;  
  69.         for ( double  d : TempDB.mapMonthSaleMoney.values()){  
  70.             group += d;  
  71.         }  
  72.         double  prize = group *  0.01 ;  
  73.         System.out.println(user+"当月团队业务奖金" +prize);  
  74.         return  prize;  
  75.     }  
  76. }  
/**
 * 计算奖金的对象
 */
public class Prize {
	/**
	 * 计算某人在某段时间内的奖金,有些参数在演示中并不会使用,
	 * 但是在实际业务实现上是会用的,为了表示这是个具体的业务方法,
	 * 因此这些参数被保留了
	 * @param user 被计算奖金的人员
	 * @param begin 计算奖金的开始时间
	 * @param end 计算奖金的结束时间
	 * @return 某人在某段时间内的奖金
	 */
	public  double calcPrize(String user,Date begin,Date end){
		double prize = 0.0;	
		//计算当月业务奖金,所有人都会计算
		prize = this.monthPrize(user, begin, end);
		//计算累计奖金
		prize += this.sumPrize(user, begin, end);
		
		//需要判断该人员是普通人员还是业务经理,团队奖金只有业务经理才有
		if(this.isManager(user)){
			prize += this.groupPrize(user, begin, end);
		}
		return prize;
	}

	/**
	 * 计算某人的当月业务奖金,参数重复,就不再注释了
	 */
	private double monthPrize(String user, Date begin, Date end) {
		//计算当月业务奖金,按照人员去获取当月的业务额,然后再乘以3%
		double prize = TempDB.mapMonthSaleMoney.get(user) * 0.03;
		System.out.println(user+"当月业务奖金"+prize);
		return prize;
	}

	/**
	 * 计算某人的累计奖金,参数重复,就不再注释了
	 */
	public double sumPrize(String user, Date begin, Date end) {
		//计算累计奖金,其实应该按照人员去获取累计的业务额,然后再乘以0.1%
		//简单演示一下,假定大家的累计业务额都是1000000元
		double prize = 1000000 * 0.001;
		System.out.println(user+"累计奖金"+prize);
		return prize;
	}	

	/**
	 * 判断人员是普通人员还是业务经理
	 * @param user 被判断的人员
	 * @return true表示是业务经理,false表示是普通人员
	 */
	private boolean isManager(String user){
		//应该从数据库中获取人员对应的职务
		//为了演示,简单点判断,只有王五是经理
		if("王五".equals(user)){
			return true;			
		}
		return false;
	}
	/**
	 * 计算当月团队业务奖,参数重复,就不再注释了
	 */
	public double groupPrize(String user, Date begin, Date end) {
		//计算当月团队业务奖金,先计算出团队总的业务额,然后再乘以1%,
//假设都是一个团队的
		double group = 0.0;
		for(double d : TempDB.mapMonthSaleMoney.values()){
			group += d;
		}
		double prize = group * 0.01;
		System.out.println(user+"当月团队业务奖金"+prize);
		return prize;
	}
}

 (3)写个客户端来测试一下,看看是否能正确地计算奖金,示例代码如下:

Java代码
  1. public   class  Client {  
  2.     public   static   void  main(String[] args) {  
  3.         //先创建计算奖金的对象   
  4.         Prize p = new  Prize();  
  5.           
  6.         //日期对象都没有用上,所以传null就可以了   
  7.         double  zs = p.calcPrize( "张三" , null , null );          
  8.         System.out.println("==========张三应得奖金:" +zs);  
  9.         double  ls = p.calcPrize( "李四" , null , null );  
  10.         System.out.println("==========李四应得奖金:" +ls);       
  11.         double  ww = p.calcPrize( "王五" , null , null );  
  12.         System.out.println("==========王经理应得奖金:" +ww);  
  13.     }  
  14. }  
public class Client {
	public static void main(String[] args) {
		//先创建计算奖金的对象
		Prize p = new Prize();
		
		//日期对象都没有用上,所以传null就可以了
		double zs = p.calcPrize("张三",null,null);		
		System.out.println("==========张三应得奖金:"+zs);
		double ls = p.calcPrize("李四",null,null);
		System.out.println("==========李四应得奖金:"+ls);		
		double ww = p.calcPrize("王五",null,null);
		System.out.println("==========王经理应得奖金:"+ww);
	}
}

 

 测试运行的结果如下: 

 

Java代码
  1. 张三当月业务奖金 300.0   
  2. 张三累计奖金1000.0   
  3. ==========张三应得奖金:1300.0   
  4. 李四当月业务奖金600.0   
  5. 李四累计奖金1000.0   
  6. ==========李四应得奖金:1600.0   
  7. 王五当月业务奖金900.0   
  8. 王五累计奖金1000.0   
  9. 王五当月团队业务奖金600.0   
  10. ==========王经理应得奖金:2500.0   
张三当月业务奖金300.0
张三累计奖金1000.0
==========张三应得奖金:1300.0
李四当月业务奖金600.0
李四累计奖金1000.0
==========李四应得奖金:1600.0
王五当月业务奖金900.0
王五累计奖金1000.0
王五当月团队业务奖金600.0
==========王经理应得奖金:2500.0

 

 

 

1.4  有何问题 

        看了上面的实现,挺简单的嘛,就是计算方式麻烦点,每个规则都要实现。真的很简单吗?仔细想想,有没有什么问题?
        对于奖金计算,光是计算方式复杂,也就罢了,不过是实现起来会困难点,相对而言还是比较好解决的,不过是用程序把已有的算法表达出来。
        最痛苦的是,这些奖金的计算方式,经常发生变动,几乎是每个季度都会有小调整,每年都有大调整,这就要求软件的实现要足够灵活,要能够很快进行相应调整和修改,否则就不能满足实际业务的需要。
        举个简单的例子来说,现在根据业务需要,需要增加一个“环比增长奖金”,就是本月的销售额比上个月有增加,而且要达到一定的比例,当然增长比例越高,奖金 比例越大。那么软件就必须要重新实现这么个功能,并正确的添加到系统中去。过了两个月,业务奖励的策略发生了变化,不再需要这个奖金了,或者是另外换了一 个新的奖金方式了,那么软件就需要把这个功能从软件中去掉,然后再实现新的功能。
        那么上面的要求该如何实现呢?
        很明显,一种方案是通过继承来扩展功能;另外一种方案就是到计算奖金的对象里面,添加或者删除新的功能,并在计算奖金的时候,调用新的功能或是不调用某些去掉的功能,这种方案会严重违反开-闭原则。
 还有一个问题,就是在运行期间,不同人员参与的奖金计算方式也是不同的,举例来说:如果是业务经理,除了参与个人计算部分外,还要参加团队奖金的计算,这就意味着需要在运行期间动态来组合需要计算的部分,也就是会有一堆的if-else。
        总结一下,奖金计算面临如下问题:

  • (1)计算逻辑复杂
  • (2)要有足够灵活性,可以方便的增加或者减少功能
  • (3)要能动态的组合计算方式,不同的人参与的计算不同

       上面描述的奖金计算的问题,绝对没有任何夸大成分,相反已经简化不少了,还有更多麻烦没有写上来,毕竟我们的重点在设计模式,而不是业务。
       把上面的问题抽象一下,设若有一个计算奖金的对象,现在需要能够灵活的给它增加和减少功能,还需要能够动态的组合功能,每个功能就相当于在计算奖金的某个部分。
       现在的问题就是:如何才能够透明的给一个对象增加功能,并实现功能的动态组合呢?

 

 

 

未完待续

 

转载自:http://chjavach.iteye.com/blog/767739

分享到:
评论

相关推荐

    研磨设计模式源码

    《研磨设计模式源码》是一份非常宝贵的资源,它提供了设计模式的实践代码,帮助开发者深入理解并应用这些模式。设计模式是软件工程中经过长期实践总结出来的一套通用解决方案,它们描述了在特定场景下如何解决常见...

    研磨设计模式-配套源代码 UTF-8格式

    《研磨设计模式》是一本深入探讨软件设计原则与实践的经典书籍,其配套源代码提供了丰富的实例,帮助读者更好地理解和应用各种设计模式。这个UTF-8格式的压缩包包含了书中介绍的各种设计模式的实现,是学习和研究...

    研磨设计模式(完整带书签).part2.pdf

    《研磨设计模式》完整覆盖GoF讲述的23个设计模式并加以细细研磨。初级内容从基本讲起,包括每个模式的定义、功能、思路、结构、基本实现、运行调用顺序、基本应用示例等,让读者能系统、完整、准确地掌握每个模式,...

    研磨设计模式-配套源代码

    以上只是设计模式中的一部分,研磨设计模式的配套源代码可能涵盖了这些或更多的模式。通过学习这些源代码,你可以更深入地理解每个模式的实现细节,以及如何在实际项目中灵活应用。这将有助于提升你的编程技能,使...

    研磨设计模式PDF

    《研磨设计模式》这本书是陈臣和王斌两位作者合作的成果,专注于讲解软件设计中的模式应用。设计模式是软件工程中的一种最佳实践,它总结了在特定上下文中解决问题的常见方法,使得开发者可以复用这些解决方案,提高...

    研磨设计模式-配套源代码.7z

    《研磨设计模式》是一本深入探讨软件设计模式的书籍,其配套源代码包含了许多经典设计模式的实际应用示例。这些源代码可以帮助读者更直观地理解设计模式的原理和使用方法,进一步提升软件开发能力。 设计模式是软件...

    研磨设计模式[书签]

    《研磨设计模式》是由陈臣和王斌合著,由清华大学出版社出版的一本深入探讨设计模式的专业书籍。设计模式是软件工程中的一个重要概念,它代表了在特定上下文中解决问题的常见方法,经过时间和实践的验证,具有很高的...

    研磨设计模式-配套源代码.rar

    《研磨设计模式》是一本深入探讨软件设计模式的书籍,配套源代码是作者为了帮助读者更好地理解和应用书中介绍的设计模式而提供的实践示例。设计模式是软件开发中经过实践检验的、解决常见问题的模板,它为软件设计...

    研磨设计模式讲课PPT

    研磨设计模式是一本深入探讨软件设计原则与实践的书籍,其讲课PPT为我们提供了丰富的设计模式知识。设计模式是软件工程中经过实践验证的、解决常见问题的模板,是经验丰富的开发人员智慧的结晶。这些模式可以帮助...

    研磨设计模式.rar

    《研磨设计模式》这本书是软件开发领域中的经典之作,主要关注的是面向对象设计中的设计模式。设计模式是在特定上下文中解决常见问题的最佳实践,它为开发者提供了在类似情况下重复使用解决方案的模板,有助于提高...

    研磨设计模式博文集

    这个“研磨设计模式博文集”显然是一份深入探讨设计模式的资料集合,其中可能包含了对多种设计模式的详细解析、示例代码以及实际应用中的经验分享。在软件开发中,设计模式能够帮助开发者提高代码质量、可读性和可...

    研磨设计模式全部源代码

    这个压缩包“研磨设计模式全部源代码”包含了多种设计模式的实现,这些模式可以帮助开发者写出更可维护、可扩展和可复用的代码。下面将详细讲解其中可能包含的一些重要设计模式及其应用。 1. 工厂模式:这是最简单...

    研磨设计模式 源代码

    《研磨设计模式》是一本深入探讨软件设计模式的经典书籍,源代码包含了书中所讲解的各种设计模式的实际应用示例。设计模式是软件工程中的重要概念,它们是经过反复验证、在特定情境下解决常见问题的有效解决方案。...

    研磨设计模式(完整带书签).part1.pdf

    《研磨设计模式》完整覆盖GoF讲述的23个设计模式并加以细细研磨。初级内容从基本讲起,包括每个模式的定义、功能、思路、结构、基本实现、运行调用顺序、基本应用示例等,让读者能系统、完整、准确地掌握每个模式,...

    研磨设计模式 演示源代码

    "研磨设计模式 演示源代码"这个资源包含了对设计模式的详细解释和实例分析,旨在帮助学习者深入理解和应用这些模式。 1. **单例模式**:确保一个类只有一个实例,并提供一个全局访问点。在资源管理、缓存或者线程池...

    研磨设计模式-part2

    《研磨设计模式》完整覆盖GoF讲述的23个设计模式并加以细细研磨。初级内容从基本讲起,包括每个模式的定义、功能、思路、结构、基本实现、运行调用顺序、基本应用示例等,让读者能系统、完整、准确地掌握每个模式,...

    研磨设计模式-part4

    《研磨设计模式》完整覆盖GoF讲述的23个设计模式并加以细细研磨。初级内容从基本讲起,包括每个模式的定义、功能、思路、结构、基本实现、运行调用顺序、基本应用示例等,让读者能系统、完整、准确地掌握每个模式,...

    研磨设计模式-part3

    《研磨设计模式》完整覆盖GoF讲述的23个设计模式并加以细细研磨。初级内容从基本讲起,包括每个模式的定义、功能、思路、结构、基本实现、运行调用顺序、基本应用示例等,让读者能系统、完整、准确地掌握每个模式,...

    研磨设计模式视频课程PPT

    这个名为“研磨设计模式视频课程PPT”的压缩包包含了一份关于23种核心设计模式的详细教学资料,旨在帮助开发者提升软件设计的效率和可维护性。下面将对这些设计模式进行深入解析。 1. **单例模式(Singleton)**:...

    X-gen PPT下载——《研磨设计模式》 实战

    《研磨设计模式》实战是IT领域中关于软件设计的一份重要资料,它主要探讨了设计模式在实际项目中的应用。设计模式是软件工程中经过长期实践总结出的通用问题解决方案,是解决常见设计问题的经验总结。这份PPT可能是...

Global site tag (gtag.js) - Google Analytics