Java中的装饰器模式是一种结构型设计模式,它允许你在不修改现有类的情况下,动态地将新功能添加到对象上。装饰器模式通过创建一个包装了原始对象的装饰器类来实现这一点。装饰器类与原始类具有相同的接口,因此它们可以互换使用。
下面是一个简单的Java装饰器模式的实现:

public interface Coffee {
double getCost();
String getIngredients();
}public class SimpleCoffee implements Coffee {
@Override
public double getCost() {
return 1;
}
@Override
public String getIngredients() {
return "Coffee";
}
}public abstract class CoffeeDecorator implements Coffee {
protected final Coffee decoratedCoffee;
public CoffeeDecorator(Coffee coffeeToDecorate) {
this.decoratedCoffee = coffeeToDecorate;
}
@Override
public double getCost() {
return decoratedCoffee.getCost();
}
@Override
public String getIngredients() {
return decoratedCoffee.getIngredients();
}
}public class MilkCoffee extends CoffeeDecorator {
public MilkCoffee(Coffee coffeeToDecorate) {
super(coffeeToDecorate);
}
@Override
public double getCost() {
return decoratedCoffee.getCost() + 0.5;
}
@Override
public String getIngredients() {
return decoratedCoffee.getIngredients() + ", Milk";
}
}
public class WhipCoffee extends CoffeeDecorator {
public WhipCoffee(Coffee coffeeToDecorate) {
super(coffeeToDecorate);
}
@Override
public double getCost() {
return decoratedCoffee.getCost() + 0.7;
}
@Override
public String getIngredients() {
return decoratedCoffee.getIngredients() + ", Whip";
}
}public class CoffeeShop {
public static void main(String[] args) {
Coffee simpleCoffee = new SimpleCoffee();
System.out.println("Cost: " + simpleCoffee.getCost() + "; Ingredients: " + simpleCoffee.getIngredients());
Coffee milkCoffee = new MilkCoffee(simpleCoffee);
System.out.println("Cost: " + milkCoffee.getCost() + "; Ingredients: " + milkCoffee.getIngredients());
Coffee milkWhipCoffee = new WhipCoffee(milkCoffee);
System.out.println("Cost: " + milkWhipCoffee.getCost() + "; Ingredients: " + milkWhipCoffee.getIngredients());
}
}运行上面的客户端代码,你会看到不同的咖啡组合和它们的价格以及配料。
有时候,装饰器可能只针对某个特定的组件类有用。在这种情况下,可以将装饰器类作为组件类的内部类来实现。
public class Component {
public void operation() {
// ...
}
public class Decorator extends Component {
@Override
public void operation() {
super.operation();
// Additional functionality
}
}
}然后客户端可以这样使用:
Component component = new Component();
Component decoratedComponent = component.new Decorator();
decoratedComponent.operation();这种方式不太常见,因为它将装饰器和被装饰的组件紧密耦合在一起。然而,在某些情况下,如果装饰器的逻辑与被装饰的组件紧密相关,并且不打算与其他组件共享,这种方式可能是合适的。
在Java 8及更高版本中,可以利用函数式接口和Lambda表达式来更简洁地实现装饰器模式。例如,可以定义一个函数式接口来表示组件的操作:
@FunctionalInterface
public interface ComponentOperation {
void execute();
}然后创建一个具体的组件实现:
public class ConcreteComponent implements ComponentOperation {
@Override
public void execute() {
System.out.println("Executing operation in ConcreteComponent");
}
}现在,可以创建一个装饰器类,它接受一个ComponentOperation并添加额外的行为:
public class Decorator implements ComponentOperation {
private final ComponentOperation componentOperation;
private final Runnable additionalBehavior;
public Decorator(ComponentOperation componentOperation, Runnable additionalBehavior) {
this.componentOperation = componentOperation;
this.additionalBehavior = additionalBehavior;
}
@Override
public void execute() {
additionalBehavior.run(); // This could be before or after the component operation
componentOperation.execute();
// additionalBehavior.run(); // Optionally call additional behavior after the operation
}
}客户端代码可以像这样使用装饰器:
public class Client {
public static void main(String[] args) {
ComponentOperation component = new ConcreteComponent();
Runnable additionalBehavior = () -> System.out.println("Executing additional behavior");
ComponentOperation decoratedComponent = new Decorator(component, additionalBehavior);
decoratedComponent.execute();
}
}在这个例子中,我们没有使用继承或抽象类,而是使用了组合和Java 8的函数式接口来实现装饰器模式。这种方式更加灵活,并且允许在运行时动态地添加不同的行为。
装饰器模式是一种强大的设计工具,它允许开发人员在不修改现有类的情况下动态地扩展对象的行为。通过合理地使用装饰器模式,可以构建出更加灵活、可扩展和可维护的软件系统。
术因分享而日新,每获新知,喜溢心扉。 诚邀关注公众号 『
码到三十五』 ,获取更多技术资料。