2 解决方案
2.1 装饰模式来解决
用来解决上述问题的一个合理的解决方案,就是使用装饰模式。那么什么是装饰模式呢?
(1)装饰模式定义
(2)应用装饰模式来解决的思路
虽然经过简化,业务简单了很多,但是需要解决的问题不会少,还是要解决:要透明的给一个对象增加功能,并实现功能的动态组合。
所谓透明的给一个对象增加功能,换句话说就是要给一个对象增加功能,但是不能让这个对象知道,也就是不能去改动这个对象。而实现了能够给一个对象透明的增加功能,自然就能够实现功能的动态组合,比如原来的对象有A功能,现在透明的给它增加了一个B功能,是不是就相当于动态组合了A和B功能呢。
要想实现透明的给一个对象增加功能,也就是要扩展对象的功能了,使用继承啊,有人马上提出了一个方案,但很快就被否决了,那要减少或者修改功能呢?事实上继承是非常不灵活的复用方式。那就用“对象组合”嘛,又有人提出新的方案来了,这个方案得到了大家的赞同。
在装饰模式的实现中,为了能够和原来使用被装饰对象的代码实现无缝结合,是通过定义一个抽象类,让这个类实现与被装饰对象相同的接口,然后在具体实现类里面,转调被装饰的对象,在转调的前后添加新的功能,这就实现了给被装饰对象增加功能,这个思路跟“对象组合”非常类似。如果对“对象组合”不熟悉,请参见3.1的第2小节。
在转调的时候,如果觉得被装饰的对象的功能不再需要了,还可以直接替换掉,也就是不再转调,而是在装饰对象里面完全全新的实现。
2.2 模式结构和说明
装饰模式的结构如图1所示:
图1 装饰模式结构图
Component:
组件对象的接口,可以给这些对象动态的添加职责。
ConcreteComponent:
具体的组件对象,实现组件对象接口,通常就是被装饰器装饰的原始对象,也就是可以给这个对象添加职责。
Decorator:
所有装饰器的抽象父类,需要定义一个与组件接口一致的接口,并持有一个Component对象,其实就是持有一个被装饰的对象。
注意这个被装饰的对象不一定是最原始的那个对象了,也可能是被其它装饰器装饰过后的对象,反正都是实现的同一个接口,也就是同一类型。
ConcreteDecorator:
实际的装饰器对象,实现具体要向被装饰对象添加的功能。
2.3 装饰模式示例代码
(1)先来看看组件对象的接口定义,示例代码如下:
-
-
-
- public abstract class Component {
-
-
-
- public abstract void operation();
- }
(2)定义了接口,那就看看具体组件实现对象示意吧,示例代码如下:
-
-
-
- public class ConcreteComponent extends Component {
- public void operation() {
-
- }
- }
(3)接下来看看抽象的装饰器对象,示例代码如下:
-
-
-
- public abstract class Decorator extends Component {
-
-
-
- protected Component component;
-
-
-
-
- public Decorator(Component component) {
- this.component = component;
- }
- public void operation() {
-
- component.operation();
- }
- }
(4)该来看看具体的装饰器实现对象了,这里有两个示意对象,一个示意了添加状态,一个示意了添加职责。先看添加了状态的示意对象吧,示例代码如下:
-
-
-
- public class ConcreteDecoratorA extends Decorator {
- public ConcreteDecoratorA(Component component) {
- super(component);
- }
-
-
-
- private String addedState;
- public String getAddedState() {
- return addedState;
- }
- public void setAddedState(String addedState) {
- this.addedState = addedState;
- }
- public void operation() {
-
-
- super.operation();
- }
- }
接下来看看添加职责的示意对象,示例代码如下:
-
-
-
- public class ConcreteDecoratorB extends Decorator {
- public ConcreteDecoratorB(Component component) {
- super(component);
- }
-
-
-
- private void addedBehavior() {
-
- }
- public void operation() {
-
- super.operation();
- addedBehavior();
- }
- }
2.4 使用装饰模式重写示例
看完了装饰模式的基本知识,该来考虑如何使用装饰模式重写前面的示例了。要使用装饰模式来重写前面的示例,大致会有如下改变:
- 首先需要定义一个组件对象的接口,在这个接口里面定义计算奖金的业务方法,因为外部就是使用这个接口来操作装饰模式构成的对象结构中的对象
- 需要添加一个基本的实现组件接口的对象,可以让它返回奖金为0就可以了
- 把各个计算奖金的规则当作装饰器对象,需要为它们定义一个统一的抽象的装饰器对象,好约束各个具体的装饰器的接口
- 把各个计算奖金的规则实现成为具体的装饰器对象
先看看现在示例的整体结构,好整体理解和把握示例,如图2所示:
图2 使用装饰模式重写示例的程序结构示意图
(1)计算奖金的组件接口和基本的实现对象
在计算奖金的组件接口中,需要定义原本的业务方法,也就是实现奖金计算的方法,示例代码如下:
-
-
-
- public abstract class Component {
-
-
-
-
-
-
-
-
-
- public abstract double calcPrize(String user
- ,Date begin,Date end);
- }
为这个业务接口提供一个基本的实现,示例代码如下:
-
-
-
- public class ConcreteComponent extends Component{
- public double calcPrize(String user, Date begin, Date end) {
-
- return 0;
- }
- }
(2)定义抽象的装饰器
在进一步定义装饰器之前,先定义出各个装饰器公共的父类,在这里定义所有装饰器对象需要实现的方法。这个父类应该实现组件的接口,这样才能保证装饰后的对象仍然可以继续被装饰。示例代码如下:
-
-
-
- public abstract class Decorator extends Component{
-
-
-
- protected Component c;
-
-
-
-
- public Decorator(Component c){
- this.c = c;
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- return c.calcPrize(user, begin, end);
- }
- }
(3)定义一系列的装饰器对象
用一个具体的装饰器对象,来实现一条计算奖金的规则,现在有三条计算奖金的规则,那就对应有三个装饰器对象来实现,依次来看看它们的实现。
这些装饰器涉及到的TempDB跟以前一样,这里就不去赘述了。
首先来看实现计算当月业务奖金的装饰器,示例代码如下:
-
-
-
- public class MonthPrizeDecorator extends Decorator{
- public MonthPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
- double prize = TempDB.mapMonthSaleMoney.get(user) * 0.03;
- System.out.println(user+"当月业务奖金"+prize);
- return money + prize;
- }
- }
接下来看实现计算累计奖金的装饰器,示例代码如下:
-
-
-
- public class SumPrizeDecorator extends Decorator{
- public SumPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
-
- double prize = 1000000 * 0.001;
- System.out.println(user+"累计奖金"+prize);
- return money + prize;
- }
- }
接下来看实现计算当月团队业务奖金的装饰器,示例代码如下:
-
-
-
- public class GroupPrizeDecorator extends Decorator{
- public GroupPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
-
- double group = 0.0;
- for(double d : TempDB.mapMonthSaleMoney.values()){
- group += d;
- }
- double prize = group * 0.01;
- System.out.println(user+"当月团队业务奖金"+prize);
- return money + prize;
- }
- }
(4)使用装饰器的客户端
使用装饰器的客户端,首先需要创建被装饰的对象,然后创建需要的装饰对象,接下来重要的工作就是组合装饰对象,依次对前面的对象进行装饰。
有很多类似的例子,比如生活中的装修,就拿装饰墙壁来说吧:没有装饰前是原始的砖墙,这就好比是被装饰的对象,首先需要刷腻子,把墙找平,这就好比对原始的砖墙进行了一次装饰,而刷的腻子就好比是一个装饰器对象;好了,装饰一回了,接下来该刷墙面漆了,这又好比装饰了一回,刷的墙面漆就好比是又一个装饰器对象,而且这回被装饰的对象不是原始的砖墙了,而是被腻子装饰器装饰过后的墙面,也就是说后面的装饰器是在前面的装饰器装饰过后的基础之上,继续装饰的,类似于一层一层叠加的功能。
同样的道理,计算奖金也是这样,先创建基本的奖金对象,然后组合需要计算的奖金类型,依次组合计算,最后的结果就是总的奖金。示例代码如下:
-
-
-
- public class Client {
- public static void main(String[] args) {
-
- Component c1 = new ConcreteComponent();
-
-
-
-
-
-
- Decorator d1 = new MonthPrizeDecorator(c1);
- Decorator d2 = new SumPrizeDecorator(d1);
-
-
-
- double zs = d2.calcPrize("张三",null,null);
- System.out.println("==========张三应得奖金:"+zs);
- double ls = d2.calcPrize("李四",null,null);
- System.out.println("==========李四应得奖金:"+ls);
-
-
- Decorator d3 = new GroupPrizeDecorator(d2);
- double ww = d3.calcPrize("王五",null,null);
- System.out.println("==========王经理应得奖金:"+ww);
- }
- }
测试一下,看看结果,示例如下:
- 张三当月业务奖金300.0
- 张三累计奖金1000.0
- ==========张三应得奖金:1300.0
- 李四当月业务奖金600.0
- 李四累计奖金1000.0
- ==========李四应得奖金:1600.0
- 王五当月业务奖金900.0
- 王五累计奖金1000.0
- 王五当月团队业务奖金600.0
- ==========王经理应得奖金:2500.0
当测试运行的时候会按照装饰器的组合顺序,依次调用相应的装饰器来执行业务功能,是一个递归的调用方法,以业务经理“王五”的奖金计算做例子,画个图来说明奖金的计算过程吧,看看是如何调用的,如图3所示:
图3 装饰模式示例的组合和调用过程示意图
这个图很好的揭示了装饰模式的组合和调用过程,请仔细体会一下。
如同上面的示例,对于基本的计算奖金的对象而言,由于计算奖金的逻辑太过于复杂,而且需要在不同的情况下进行不同的运算,为了灵活性,把多种计算奖金的方式分散到不同的装饰器对象里面,采用动态组合的方式,来给基本的计算奖金的对象增添计算奖金的功能,每个装饰器相当于计算奖金的一个部分。
这种方式明显比为基本的计算奖金的对象增加子类来得更灵活,因为装饰模式的起源点是采用对象组合的方式,然后在组合的时候顺便增加些功能。为了达到一层一层组装的效果,装饰模式还要求装饰器要实现与被装饰对象相同的业务接口,这样才能以同一种方式依次组合下去。
灵活性还体现在动态上,如果是继承的方式,那么所有的类实例都有这个功能了,而采用装饰模式,可以动态的为某几个对象实例添加功能,而不是对整个类添加功能。比如上面示例中,客户端测试的时候,对张三李四就只是组合了两个功能,对王五就组合了三个功能,但是原始的计算奖金的类都是一样的,只是动态的为它增加的功能不同而已。
2 解决方案
2.1 装饰模式来解决
用来解决上述问题的一个合理的解决方案,就是使用装饰模式。那么什么是装饰模式呢?
(1)装饰模式定义
(2)应用装饰模式来解决的思路
虽然经过简化,业务简单了很多,但是需要解决的问题不会少,还是要解决:要透明的给一个对象增加功能,并实现功能的动态组合。
所谓透明的给一个对象增加功能,换句话说就是要给一个对象增加功能,但是不能让这个对象知道,也就是不能去改动这个对象。而实现了能够给一个对象透明的增加功能,自然就能够实现功能的动态组合,比如原来的对象有A功能,现在透明的给它增加了一个B功能,是不是就相当于动态组合了A和B功能呢。
要想实现透明的给一个对象增加功能,也就是要扩展对象的功能了,使用继承啊,有人马上提出了一个方案,但很快就被否决了,那要减少或者修改功能呢?事实上继承是非常不灵活的复用方式。那就用“对象组合”嘛,又有人提出新的方案来了,这个方案得到了大家的赞同。
在装饰模式的实现中,为了能够和原来使用被装饰对象的代码实现无缝结合,是通过定义一个抽象类,让这个类实现与被装饰对象相同的接口,然后在具体实现类里面,转调被装饰的对象,在转调的前后添加新的功能,这就实现了给被装饰对象增加功能,这个思路跟“对象组合”非常类似。如果对“对象组合”不熟悉,请参见3.1的第2小节。
在转调的时候,如果觉得被装饰的对象的功能不再需要了,还可以直接替换掉,也就是不再转调,而是在装饰对象里面完全全新的实现。
2.2 模式结构和说明
装饰模式的结构如图1所示:
图1 装饰模式结构图
Component:
组件对象的接口,可以给这些对象动态的添加职责。
ConcreteComponent:
具体的组件对象,实现组件对象接口,通常就是被装饰器装饰的原始对象,也就是可以给这个对象添加职责。
Decorator:
所有装饰器的抽象父类,需要定义一个与组件接口一致的接口,并持有一个Component对象,其实就是持有一个被装饰的对象。
注意这个被装饰的对象不一定是最原始的那个对象了,也可能是被其它装饰器装饰过后的对象,反正都是实现的同一个接口,也就是同一类型。
ConcreteDecorator:
实际的装饰器对象,实现具体要向被装饰对象添加的功能。
2.3 装饰模式示例代码
(1)先来看看组件对象的接口定义,示例代码如下:
-
-
-
- public abstract class Component {
-
-
-
- public abstract void operation();
- }
(2)定义了接口,那就看看具体组件实现对象示意吧,示例代码如下:
-
-
-
- public class ConcreteComponent extends Component {
- public void operation() {
-
- }
- }
(3)接下来看看抽象的装饰器对象,示例代码如下:
-
-
-
- public abstract class Decorator extends Component {
-
-
-
- protected Component component;
-
-
-
-
- public Decorator(Component component) {
- this.component = component;
- }
- public void operation() {
-
- component.operation();
- }
- }
(4)该来看看具体的装饰器实现对象了,这里有两个示意对象,一个示意了添加状态,一个示意了添加职责。先看添加了状态的示意对象吧,示例代码如下:
-
-
-
- public class ConcreteDecoratorA extends Decorator {
- public ConcreteDecoratorA(Component component) {
- super(component);
- }
-
-
-
- private String addedState;
- public String getAddedState() {
- return addedState;
- }
- public void setAddedState(String addedState) {
- this.addedState = addedState;
- }
- public void operation() {
-
-
- super.operation();
- }
- }
接下来看看添加职责的示意对象,示例代码如下:
-
-
-
- public class ConcreteDecoratorB extends Decorator {
- public ConcreteDecoratorB(Component component) {
- super(component);
- }
-
-
-
- private void addedBehavior() {
-
- }
- public void operation() {
-
- super.operation();
- addedBehavior();
- }
- }
2.4 使用装饰模式重写示例
看完了装饰模式的基本知识,该来考虑如何使用装饰模式重写前面的示例了。要使用装饰模式来重写前面的示例,大致会有如下改变:
- 首先需要定义一个组件对象的接口,在这个接口里面定义计算奖金的业务方法,因为外部就是使用这个接口来操作装饰模式构成的对象结构中的对象
- 需要添加一个基本的实现组件接口的对象,可以让它返回奖金为0就可以了
- 把各个计算奖金的规则当作装饰器对象,需要为它们定义一个统一的抽象的装饰器对象,好约束各个具体的装饰器的接口
- 把各个计算奖金的规则实现成为具体的装饰器对象
先看看现在示例的整体结构,好整体理解和把握示例,如图2所示:
图2 使用装饰模式重写示例的程序结构示意图
(1)计算奖金的组件接口和基本的实现对象
在计算奖金的组件接口中,需要定义原本的业务方法,也就是实现奖金计算的方法,示例代码如下:
-
-
-
- public abstract class Component {
-
-
-
-
-
-
-
-
-
- public abstract double calcPrize(String user
- ,Date begin,Date end);
- }
为这个业务接口提供一个基本的实现,示例代码如下:
-
-
-
- public class ConcreteComponent extends Component{
- public double calcPrize(String user, Date begin, Date end) {
-
- return 0;
- }
- }
(2)定义抽象的装饰器
在进一步定义装饰器之前,先定义出各个装饰器公共的父类,在这里定义所有装饰器对象需要实现的方法。这个父类应该实现组件的接口,这样才能保证装饰后的对象仍然可以继续被装饰。示例代码如下:
-
-
-
- public abstract class Decorator extends Component{
-
-
-
- protected Component c;
-
-
-
-
- public Decorator(Component c){
- this.c = c;
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- return c.calcPrize(user, begin, end);
- }
- }
(3)定义一系列的装饰器对象
用一个具体的装饰器对象,来实现一条计算奖金的规则,现在有三条计算奖金的规则,那就对应有三个装饰器对象来实现,依次来看看它们的实现。
这些装饰器涉及到的TempDB跟以前一样,这里就不去赘述了。
首先来看实现计算当月业务奖金的装饰器,示例代码如下:
-
-
-
- public class MonthPrizeDecorator extends Decorator{
- public MonthPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
- double prize = TempDB.mapMonthSaleMoney.get(user) * 0.03;
- System.out.println(user+"当月业务奖金"+prize);
- return money + prize;
- }
- }
接下来看实现计算累计奖金的装饰器,示例代码如下:
-
-
-
- public class SumPrizeDecorator extends Decorator{
- public SumPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
-
- double prize = 1000000 * 0.001;
- System.out.println(user+"累计奖金"+prize);
- return money + prize;
- }
- }
接下来看实现计算当月团队业务奖金的装饰器,示例代码如下:
-
-
-
- public class GroupPrizeDecorator extends Decorator{
- public GroupPrizeDecorator(Component c){
- super(c);
- }
- public double calcPrize(String user, Date begin, Date end) {
-
- double money = super.calcPrize(user, begin, end);
-
-
- double group = 0.0;
- for(double d : TempDB.mapMonthSaleMoney.values()){
- group += d;
- }
- double prize = group * 0.01;
- System.out.println(user+"当月团队业务奖金"+prize);
- return money + prize;
- }
- }
(4)使用装饰器的客户端
使用装饰器的客户端,首先需要创建被装饰的对象,然后创建需要的装饰对象,接下来重要的工作就是组合装饰对象,依次对前面的对象进行装饰。
有很多类似的例子,比如生活中的装修,就拿装饰墙壁来说吧:没有装饰前是原始的砖墙,这就好比是被装饰的对象,首先需要刷腻子,把墙找平,这就好比对原始的砖墙进行了一次装饰,而刷的腻子就好比是一个装饰器对象;好了,装饰一回了,接下来该刷墙面漆了,这又好比装饰了一回,刷的墙面漆就好比是又一个装饰器对象,而且这回被装饰的对象不是原始的砖墙了,而是被腻子装饰器装饰过后的墙面,也就是说后面的装饰器是在前面的装饰器装饰过后的基础之上,继续装饰的,类似于一层一层叠加的功能。
同样的道理,计算奖金也是这样,先创建基本的奖金对象,然后组合需要计算的奖金类型,依次组合计算,最后的结果就是总的奖金。示例代码如下:
-
-
-
- public class Client {
- public static void main(String[] args) {
-
- Component c1 = new ConcreteComponent();
-
-
-
-
-
-
- Decorator d1 = new MonthPrizeDecorator(c1);
- Decorator d2 = new SumPrizeDecorator(d1);
-
-
-
- double zs = d2.calcPrize("张三",null,null);
- System.out.println("==========张三应得奖金:"+zs);
- double ls = d2.calcPrize("李四",null,null);
- System.out.println("==========李四应得奖金:"+ls);
-
-
- Decorator d3 = new GroupPrizeDecorator(d2);
- double ww = d3.calcPrize("王五",null,null);
- System.out.println("==========王经理应得奖金:"+ww);
- }
- }
测试一下,看看结果,示例如下:
- 张三当月业务奖金300.0
- 张三累计奖金1000.0
- ==========张三应得奖金:1300.0
- 李四当月业务奖金600.0
- 李四累计奖金1000.0
- ==========李四应得奖金:1600.0
- 王五当月业务奖金900.0
- 王五累计奖金1000.0
- 王五当月团队业务奖金600.0
- ==========王经理应得奖金:2500.0
当测试运行的时候会按照装饰器的组合顺序,依次调用相应的装饰器来执行业务功能,是一个递归的调用方法,以业务经理“王五”的奖金计算做例子,画个图来说明奖金的计算过程吧,看看是如何调用的,如图3所示:
图3 装饰模式示例的组合和调用过程示意图
这个图很好的揭示了装饰模式的组合和调用过程,请仔细体会一下。
如同上面的示例,对于基本的计算奖金的对象而言,由于计算奖金的逻辑太过于复杂,而且需要在不同的情况下进行不同的运算,为了灵活性,把多种计算奖金的方式分散到不同的装饰器对象里面,采用动态组合的方式,来给基本的计算奖金的对象增添计算奖金的功能,每个装饰器相当于计算奖金的一个部分。
这种方式明显比为基本的计算奖金的对象增加子类来得更灵活,因为装饰模式的起源点是采用对象组合的方式,然后在组合的时候顺便增加些功能。为了达到一层一层组装的效果,装饰模式还要求装饰器要实现与被装饰对象相同的业务接口,这样才能以同一种方式依次组合下去。
灵活性还体现在动态上,如果是继承的方式,那么所有的类实例都有这个功能了,而采用装饰模式,可以动态的为某几个对象实例添加功能,而不是对整个类添加功能。比如上面示例中,客户端测试的时候,对张三李四就只是组合了两个功能,对王五就组合了三个功能,但是原始的计算奖金的类都是一样的,只是动态的为它增加的功能不同而已。
注:本文转自 http://chjavach.iteye.com