138 lines
5.4 KiB
Markdown
138 lines
5.4 KiB
Markdown
|
---
|
||
|
layout: pattern
|
||
|
title: Decorator
|
||
|
folder: decorator
|
||
|
permalink: /patterns/decorator/
|
||
|
categories: Structural
|
||
|
tags:
|
||
|
- Gang Of Four
|
||
|
- Extensibility
|
||
|
---
|
||
|
|
||
|
## 或称
|
||
|
包装器
|
||
|
|
||
|
## 目的
|
||
|
动态的为对象附加额外的职责。装饰器为子类提供了灵活的替代方案,以扩展功能。
|
||
|
|
||
|
## 解释
|
||
|
|
||
|
真实世界例子
|
||
|
|
||
|
> 附近的山丘上住着一个愤怒的巨魔。通常它是徒手的,但有时它有武器。为了武装巨魔不必创建新的巨魔,而是用合适的武器动态的装饰它。
|
||
|
|
||
|
通俗的说
|
||
|
|
||
|
> 装饰者模式让你可以在运行时通过把对象包装进一个装饰类对象中来动态的改变一个对象的行为。
|
||
|
|
||
|
维基百科说
|
||
|
|
||
|
> 在面向对象的编程中,装饰器模式是一种设计模式,它允许将行为静态或动态地添加到单个对象中,而不会影响同一类中其他对象的行为。装饰器模式通常对于遵守单一责任原则很有用,因为它允许将功能划分到具有唯一关注领域的类之间。
|
||
|
|
||
|
**程序示例**
|
||
|
|
||
|
以巨魔的为例。首先我有有一个简单的巨魔,实现了巨魔接口。
|
||
|
|
||
|
```java
|
||
|
public interface Troll {
|
||
|
void attack();
|
||
|
int getAttackPower();
|
||
|
void fleeBattle();
|
||
|
}
|
||
|
|
||
|
public class SimpleTroll implements Troll {
|
||
|
|
||
|
private static final Logger LOGGER = LoggerFactory.getLogger(SimpleTroll.class);
|
||
|
|
||
|
@Override
|
||
|
public void attack() {
|
||
|
LOGGER.info("The troll tries to grab you!");
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public int getAttackPower() {
|
||
|
return 10;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void fleeBattle() {
|
||
|
LOGGER.info("The troll shrieks in horror and runs away!");
|
||
|
}
|
||
|
}
|
||
|
```
|
||
|
|
||
|
下面我们想为巨魔添加球棒。我们可以用装饰者来动态的实现。
|
||
|
|
||
|
```java
|
||
|
public class ClubbedTroll implements Troll {
|
||
|
|
||
|
private static final Logger LOGGER = LoggerFactory.getLogger(ClubbedTroll.class);
|
||
|
|
||
|
private final Troll decorated;
|
||
|
|
||
|
public ClubbedTroll(Troll decorated) {
|
||
|
this.decorated = decorated;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void attack() {
|
||
|
decorated.attack();
|
||
|
LOGGER.info("The troll swings at you with a club!");
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public int getAttackPower() {
|
||
|
return decorated.getAttackPower() + 10;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void fleeBattle() {
|
||
|
decorated.fleeBattle();
|
||
|
}
|
||
|
}
|
||
|
```
|
||
|
|
||
|
这里是巨魔的实战
|
||
|
|
||
|
```java
|
||
|
// simple troll
|
||
|
var troll = new SimpleTroll();
|
||
|
troll.attack(); // The troll tries to grab you!
|
||
|
troll.fleeBattle(); // The troll shrieks in horror and runs away!
|
||
|
|
||
|
// change the behavior of the simple troll by adding a decorator
|
||
|
var clubbedTroll = new ClubbedTroll(troll);
|
||
|
clubbedTroll.attack(); // The troll tries to grab you! The troll swings at you with a club!
|
||
|
clubbedTroll.fleeBattle(); // The troll shrieks in horror and runs away!
|
||
|
```
|
||
|
|
||
|
## 类图
|
||
|

|
||
|
|
||
|
## 适用性
|
||
|
使用装饰者
|
||
|
|
||
|
* 动态透明地向单个对象添加职责,即不影响其他对象
|
||
|
* 对于可以撤销的责任
|
||
|
* 当通过子类化进行扩展是不切实际的。有时可能会有大量的独立扩展,并且会产生大量的子类来支持每种组合。 否则类定义可能被隐藏或无法用于子类化。
|
||
|
|
||
|
## 教程
|
||
|
* [Decorator Pattern Tutorial](https://www.journaldev.com/1540/decorator-design-pattern-in-java-example)
|
||
|
|
||
|
## Java世界的例子
|
||
|
* [java.io.InputStream](http://docs.oracle.com/javase/8/docs/api/java/io/InputStream.html), [java.io.OutputStream](http://docs.oracle.com/javase/8/docs/api/java/io/OutputStream.html),
|
||
|
[java.io.Reader](http://docs.oracle.com/javase/8/docs/api/java/io/Reader.html) and [java.io.Writer](http://docs.oracle.com/javase/8/docs/api/java/io/Writer.html)
|
||
|
* [java.util.Collections#synchronizedXXX()](http://docs.oracle.com/javase/8/docs/api/java/util/Collections.html#synchronizedCollection-java.util.Collection-)
|
||
|
* [java.util.Collections#unmodifiableXXX()](http://docs.oracle.com/javase/8/docs/api/java/util/Collections.html#unmodifiableCollection-java.util.Collection-)
|
||
|
* [java.util.Collections#checkedXXX()](http://docs.oracle.com/javase/8/docs/api/java/util/Collections.html#checkedCollection-java.util.Collection-java.lang.Class-)
|
||
|
|
||
|
|
||
|
## 鸣谢
|
||
|
|
||
|
* [Design Patterns: Elements of Reusable Object-Oriented Software](https://www.amazon.com/gp/product/0201633612/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=0201633612&linkCode=as2&tag=javadesignpat-20&linkId=675d49790ce11db99d90bde47f1aeb59)
|
||
|
* [Functional Programming in Java: Harnessing the Power of Java 8 Lambda Expressions](https://www.amazon.com/gp/product/1937785467/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=1937785467&linkCode=as2&tag=javadesignpat-20&linkId=7e4e2fb7a141631491534255252fd08b)
|
||
|
* [J2EE Design Patterns](https://www.amazon.com/gp/product/0596004273/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=0596004273&linkCode=as2&tag=javadesignpat-20&linkId=48d37c67fb3d845b802fa9b619ad8f31)
|
||
|
* [Head First Design Patterns: A Brain-Friendly Guide](https://www.amazon.com/gp/product/0596007124/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=0596007124&linkCode=as2&tag=javadesignpat-20&linkId=6b8b6eea86021af6c8e3cd3fc382cb5b)
|
||
|
* [Refactoring to Patterns](https://www.amazon.com/gp/product/0321213351/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=0321213351&linkCode=as2&tag=javadesignpat-20&linkId=2a76fcb387234bc71b1c61150b3cc3a7)
|
||
|
* [J2EE Design Patterns](https://www.amazon.com/gp/product/0596004273/ref=as_li_tl?ie=UTF8&camp=1789&creative=9325&creativeASIN=0596004273&linkCode=as2&tag=javadesignpat-20&linkId=f27d2644fbe5026ea448791a8ad09c94)
|