From e09cada960c0899ba6d8a9f116aa4791f2983130 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ilkka=20Sepp=C3=A4l=C3=A4?= Date: Sun, 7 Apr 2024 18:59:47 +0300 Subject: [PATCH] docs: improve decorator --- decorator/README.md | 147 ++++++++++-------- .../java/com/iluwatar/decorator/AppTest.java | 1 - .../iluwatar/decorator/SimpleTrollTest.java | 2 +- 3 files changed, 82 insertions(+), 68 deletions(-) diff --git a/decorator/README.md b/decorator/README.md index 3df72d667..c770a7fcc 100644 --- a/decorator/README.md +++ b/decorator/README.md @@ -3,100 +3,96 @@ title: Decorator category: Structural language: en tag: - - Gang of Four - - Extensibility + - Gang of Four + - Enhancement + - Extensibility + - Wrapping --- ## Also known as -Wrapper +* Smart Proxy +* Wrapper ## Intent -Attach additional responsibilities to an object dynamically. Decorators provide a flexible -alternative to subclassing for extending functionality. +The Decorator pattern allows for the dynamic addition of responsibilities to objects without modifying their existing code. It achieves this by providing a way to "wrap" objects within objects of similar interface. ## Explanation Real-world example -> There is an angry troll living in the nearby hills. Usually, it goes bare-handed but sometimes it -> has a weapon. To arm the troll it's not necessary to create a new troll but to decorate it -> dynamically with a suitable weapon. +> There is an angry troll living in the nearby hills. Usually, it goes bare-handed, but sometimes it has a weapon. To arm the troll it's not necessary to create a new troll but to decorate it dynamically with a suitable weapon. In plain words -> Decorator pattern lets you dynamically change the behavior of an object at run time by wrapping -> them in an object of a decorator class. +> Decorator pattern lets you dynamically change the behavior of an object at run time by wrapping them in an object of a decorator class. Wikipedia says -> In object-oriented programming, the decorator pattern is a design pattern that allows behavior to -> be added to an individual object, either statically or dynamically, without affecting the behavior -> of other objects from the same class. The decorator pattern is often useful for adhering to the -> Single Responsibility Principle, as it allows functionality to be divided between classes with -> unique areas of concern as well as to the Open-Closed Principle, by allowing the functionality -> of a class to be extended without being modified. +> In object-oriented programming, the decorator pattern is a design pattern that allows behavior to be added to an individual object, either statically or dynamically, without affecting the behavior of other objects from the same class. The decorator pattern is often useful for adhering to the Single Responsibility Principle, as it allows functionality to be divided between classes with unique areas of concern as well as to the Open-Closed Principle, by allowing the functionality of a class to be extended without being modified. **Programmatic Example** -Let's take the troll example. First of all we have a `SimpleTroll` implementing the `Troll` -interface: +Let's take the troll example. First of all we have a `SimpleTroll` implementing the `Troll` interface: ```java public interface Troll { - void attack(); - int getAttackPower(); - void fleeBattle(); + void attack(); + + int getAttackPower(); + + void fleeBattle(); } @Slf4j public class SimpleTroll implements Troll { - @Override - public void attack() { - LOGGER.info("The troll tries to grab you!"); - } + @Override + public void attack() { + LOGGER.info("The troll tries to grab you!"); + } - @Override - public int getAttackPower() { - return 10; - } + @Override + public int getAttackPower() { + return 10; + } - @Override - public void fleeBattle() { - LOGGER.info("The troll shrieks in horror and runs away!"); - } + @Override + public void fleeBattle() { + LOGGER.info("The troll shrieks in horror and runs away!"); + } } ``` Next, we want to add a club for the troll. We can do it dynamically by using a decorator: ```java + @Slf4j public class ClubbedTroll implements Troll { - private final Troll decorated; + private final Troll decorated; - public ClubbedTroll(Troll decorated) { - this.decorated = 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 void attack() { + decorated.attack(); + LOGGER.info("The troll swings at you with a club!"); + } - @Override - public int getAttackPower() { - return decorated.getAttackPower() + 10; - } + @Override + public int getAttackPower() { + return decorated.getAttackPower() + 10; + } - @Override - public void fleeBattle() { - decorated.fleeBattle(); - } + @Override + public void fleeBattle() { + decorated.fleeBattle(); + } } ``` @@ -105,17 +101,17 @@ Here's the troll in action: ```java // simple troll LOGGER.info("A simple looking troll approaches."); -var troll = new SimpleTroll(); +var troll=new SimpleTroll(); troll.attack(); troll.fleeBattle(); -LOGGER.info("Simple troll power: {}.\n", troll.getAttackPower()); +LOGGER.info("Simple troll power: {}.\n",troll.getAttackPower()); // change the behavior of the simple troll by adding a decorator LOGGER.info("A troll with huge club surprises you."); -var clubbedTroll = new ClubbedTroll(troll); +var clubbedTroll=new ClubbedTroll(troll); clubbedTroll.attack(); clubbedTroll.fleeBattle(); -LOGGER.info("Clubbed troll power: {}.\n", clubbedTroll.getAttackPower()); +LOGGER.info("Clubbed troll power: {}.\n",clubbedTroll.getAttackPower()); ``` Program output: @@ -124,13 +120,13 @@ Program output: A simple looking troll approaches. The troll tries to grab you! The troll shrieks in horror and runs away! -Simple troll power: 10. +Simple troll power:10. A troll with huge club surprises you. The troll tries to grab you! The troll swings at you with a club! The troll shrieks in horror and runs away! -Clubbed troll power: 20. +Clubbed troll power:20. ``` ## Class diagram @@ -141,12 +137,10 @@ Clubbed troll power: 20. Decorator is used to: -* Add responsibilities to individual objects dynamically and transparently, that is, without -affecting other objects. +* Add responsibilities to individual objects dynamically and transparently, that is, without affecting other objects. * For responsibilities that can be withdrawn. -* When extension by subclassing is impractical. Sometimes a large number of independent extensions -are possible and would produce an explosion of subclasses to support every combination. Or a class -definition may be hidden or otherwise unavailable for subclassing. +* When extending a class is impractical due to the proliferation of subclasses that could result. +* For when a class definition might be hidden or otherwise unavailable for subclassing. ## Tutorials @@ -154,12 +148,32 @@ definition may be hidden or otherwise unavailable for subclassing. ## Known uses - * [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-) +* GUI toolkits often use decorators to dynamically add behaviors like scrolling, borders, or layout management to components. +* [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-) +## Consequences + +Benefits: + +* Greater flexibility than static inheritance. +* Avoids feature-laden classes high up in the hierarchy. +* A decorator and its component aren't identical. +* Responsibilities can be added or removed at runtime. + +Trade-offs: + +* A decorator and its component aren't identical, so tests for object type will fail. +* Decorators can lead to a system with lots of small objects that look alike to the programmer, making the desired configuration hard to achieve. +* Overuse can complicate the code structure due to the introduction of numerous small classes. + +## Related Patterns + +* [Adapter](https://java-design-patterns.com/patterns/adapter/): A decorator changes an object's responsibilities, while an adapter changes an object's interface. +* [Composite](https://java-design-patterns.com/patterns/composite/): Decorators can be viewed as a degenerate composite with only one component. However, a decorator adds additional responsibilities—it isn't intended for object aggregation. +* [Strategy](https://java-design-patterns.com/patterns/strategy/): Decorator lets you change the skin of an object, while Strategy lets you change the guts. ## Credits @@ -169,3 +183,4 @@ definition may be hidden or otherwise unavailable for subclassing. * [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) +* [Pattern-Oriented Software Architecture, Volume 1: A System of Patterns](https://amzn.to/4aKFTgS) diff --git a/decorator/src/test/java/com/iluwatar/decorator/AppTest.java b/decorator/src/test/java/com/iluwatar/decorator/AppTest.java index 32609d3b9..15285dd11 100644 --- a/decorator/src/test/java/com/iluwatar/decorator/AppTest.java +++ b/decorator/src/test/java/com/iluwatar/decorator/AppTest.java @@ -35,7 +35,6 @@ class AppTest { /** * Issue: Add at least one assertion to this test case. - * * Solution: Inserted assertion to check whether the execution of the main method in {@link App#main(String[])} * throws an exception. */ diff --git a/decorator/src/test/java/com/iluwatar/decorator/SimpleTrollTest.java b/decorator/src/test/java/com/iluwatar/decorator/SimpleTrollTest.java index e0bde46db..65bde0455 100644 --- a/decorator/src/test/java/com/iluwatar/decorator/SimpleTrollTest.java +++ b/decorator/src/test/java/com/iluwatar/decorator/SimpleTrollTest.java @@ -67,7 +67,7 @@ class SimpleTrollTest { assertEquals(2, appender.getLogSize()); } - private class InMemoryAppender extends AppenderBase { + private static class InMemoryAppender extends AppenderBase { private final List log = new LinkedList<>();