装饰者模式:动态地给一个对象增加一些额外的职责,就增加的功能来说,Decorator模式相比生成子类更加灵活。
特征:
- 装饰对象和真实对象有相同的接口
- 装饰对象包含一个真实对象的引用(reference)
- 装饰对象可以在转发这些请求以前或以后增加一些附加功能
步骤 1、创建一个接口:Shape.java
public interface Shape {
void draw();
}
步骤 2、创建实现接口的实体类。
Rectangle.java
public class Rectangle implements Shape {
@Override
public void draw() {
System.out.println("Shape: Rectangle");
}
}
Circle.java
public class Circle implements Shape {
@Override
public void draw() {
System.out.println("Shape: Circle");
}
}
步骤 3、创建实现了 Shape 接口的抽象装饰类。
ShapeDecorator.java
public abstract class ShapeDecorator implements Shape {
protected Shape decoratedShape;
public ShapeDecorator(Shape decoratedShape){
this.decoratedShape = decoratedShape;
}
public void draw(){
decoratedShape.draw();
}
}
步骤 4、创建扩展了 ShapeDecorator 类的实体装饰类。
RedShapeDecorator.java
public class RedShapeDecorator extends ShapeDecorator {
public RedShapeDecorator(Shape decoratedShape) {
super(decoratedShape);
}
@Override
public void draw() {
decoratedShape.draw();
setRedBorder(decoratedShape);
}
private void setRedBorder(Shape decoratedShape){
System.out.println("Border Color: Red");
}
}
步骤 5、使用 RedShapeDecorator 来装饰 Shape 对象。
DecoratorPatternDemo.java
public class DecoratorPatternDemo {
public static void main(String[] args) {
Shape circle = new Circle();
Shape redCircle = new RedShapeDecorator(new Circle());
Shape redRectangle = new RedShapeDecorator(new Rectangle());
System.out.println("Circle with normal border");
circle.draw();
System.out.println("\nCircle of red border");
redCircle.draw();
System.out.println("\nRectangle of red border");
redRectangle.draw();
}
}
步骤 6、执行程序,输出结果:
Circle with normal border
Shape: Circle
Circle of red border
Shape: Circle
Border Color: Red
Rectangle of red border
Shape: Rectangle
Border Color: Red
由于使用装饰模式,可以比使用继承关系需要较少数目的类。使用较少的类,当然使设计比较易于进行。但是,在另一方面,使用装饰模式会产生比使用继承关系更多的对象。更多的对象会使得查错变得困难,特别是这些对象看上去都很相像。
Activity、Service、Application等都是一个Context,这里面实际上就是通过装饰者模式来实现的。
https://developer.android.google.cn/reference/android/app/Activity.html
https://developer.android.google.cn/reference/android/app/Service.html
https://developer.android.google.cn/reference/android/app/Application.html
Content.java 是一个abstract类 xref: /frameworks/base/core/java/android/content/Context.java
ContextImpl.java 是Content.java通用实现 xref: /frameworks/base/core/java/android/app/ContextImpl.java
Activity、Service、Application里面调用 startActivity() 方法,实际上是调用他们的父类 ContextWrapper里面的方法;但他们各自扩展了自己额外的职责功能
xref: /frameworks/base/core/java/android/content/ContextWrapper.java
xref: /frameworks/base/core/java/android/app/Activity.java
xref: /frameworks/base/core/java/android/app/Service.java
xref: /frameworks/base/core/java/android/app/Application.java
segmentfault: 装饰者模式
菜鸟教程:装饰器模式
百度百科:装饰器模式
博客园: 《JAVA与模式》之装饰模式