1. 데코레이터 패턴이란?
- Decorate의 의미 '꾸민다'의 뜻 그대로 어떤 객체를 꾸며주기 위해 필요한 여러 요소들을 추가해주는 패턴.
- 객체에 추가 요소를 동적으로 더할 수 있는 디자인 패턴.
- 개발 확장에 있어 base 클래스 밑에 서브 클래스를 만들어 확장시킬 때, 데코레이터 패턴을 사용하면 서브 클래스를 만들어 확장할 때보다 더 확장성 있게 구현할 수 있음.
- 기본 기능에 추가할 수 있는 많은 종류의 부가 기능에서 파생되는 다양한 조합을 동적으로 구현할 수 있는 패턴.
- 객체의 결합을 통해 기능을 동적으로 유연하게 확장할 수 있도록 해주는 패턴.
- 각각이 개별적인 객체면서 parent에선 멤버인 child 객체를 사용할 수 있고, 관련 기능들을 위임하여 실행하는 패턴.
- OCP 원칙에 따라 구성된 패턴.
cf) OCP(Open-Closed Principle) 원칙 : 클래스는 확장에는 열려 있어야 하지만 변경에는 닫혀 있어야 한다.
2. 데코레이터 패턴을 사용하는 경우
- 객체의 기능을 동적으로 확장하고자 할 때 사용
- 상속받을 클래스가 상속이 불가능 할 때 사용
- 일시적 공격력 강화, 아이템을 먹는 경우 등 이에 해당
3. 데코레이터 패턴의 장점
1) 확장성 : 객체의 결합을 통해 자식 클래스를 만들지 않고도 객체의 기능을 동적으로 유연하게 확장할 수 있음.
2) 보존성 : 기존 코드를 수정하기 않고 객체를 여러 데코레이터로 래핑하여 합성할 수 있음.
3) 단일 책임 원칙 : 다양한 행동들의 여러 변형들을 구현하는 모놀리식 클래스를 여러 개의 작은 클래스들로 나눌 수 있음.
4. 데코레이터 패턴의 단점
1) 코드의 양과 스크립트의 양이 증가한다.
2) 래퍼들의 스택에서 특정 래퍼를 제거하기가 어렵다.
3) 클래스의 종류가 많아지고 객체를 파악하기 어려워진다.
4) 데코레이터의 행동이 데코레이터 스택 내의 순서에 의존하지 않는 방식으로 데코레이터를 구현하기가 어렵다.
5. 코드 구현
1) 최상위 추상 클래스 Skill을 상속받아 MagicBall 클래스와 Shield 클래스를 각각 구현
2) 최상위 추상 클래스 Skill을 상속받는 추상 클래스 Decorator 클래스도 있음.
3) Decorator 클래스를 상속받는 FireProperty 클래스, ElectricProperty 클래스를 각각 구현.
결론 : Decorator를 이용해 구현한 클래스들을 원하는 대로 MagicBall이나 Shield 클래스 래핑하여 쓸 수 있음.
- 다이어그램
- 소스 코드(c#)
using System;
using System.Collections.Generic;
namespace StudyCSharp
{
// 최상위 추상 클래스
public abstract class Skill
{
// 멤버변수
protected string name;
protected float damage;
// 추상 메소드(객체와 데코레이터에서 구현)
public abstract string GetName();
public abstract float GetDamage();
public abstract void GetInfo();
}
// 데코레이터 클래스
public abstract class Decorator : Skill
{
protected Skill skill;
}
// 꾸며질 객체 1 - 마법 구
public class MagicBall : Skill
{
public MagicBall()
{
this.name = "볼";
this.damage = 10f;
}
public override string GetName()
{
return this.name;
}
public override float GetDamage()
{
return this.damage;
}
public override void GetInfo()
{
Console.WriteLine("스킬 명\t: " + this.name);
Console.WriteLine("데미지\t: " + this.damage);
Console.WriteLine();
}
}
// 꾸며질 객체 2 - 실드
public class Shield : Skill
{
public Shield()
{
this.name = "실드";
this.damage = 0f;
}
public override string GetName()
{
return this.name;
}
public override float GetDamage()
{
return this.damage;
}
public override void GetInfo()
{
Console.WriteLine("스킬 명\t: " + this.name);
Console.WriteLine("데미지\t: " + this.damage);
Console.WriteLine();
}
}
// 불 특성 부여
public class FireProperty : Decorator
{
public FireProperty(Skill _skill)
{
this.skill = _skill;
this.name = "파이어 " + _skill.GetName();
this.damage = 3 + _skill.GetDamage();
}
public override string GetName()
{
return this.name;
}
public override float GetDamage()
{
return this.damage;
}
public override void GetInfo()
{
Console.WriteLine("스킬 명\t: " + this.name);
Console.WriteLine("데미지\t: " + this.damage);
Console.WriteLine();
}
}
// 전기 특성 부여
public class ElectricProperty : Decorator
{
public ElectricProperty(Skill _skill)
{
this.skill = _skill;
this.name = "일렉트릭 " + _skill.GetName();
this.damage = 5 + _skill.GetDamage();
}
public override string GetName()
{
return this.name;
}
public override float GetDamage()
{
return this.damage;
}
public override void GetInfo()
{
Console.WriteLine("스킬 명\t: " + this.name);
Console.WriteLine("데미지\t: " + this.damage);
Console.WriteLine();
}
}
class Program
{
static void Main(string[] args)
{
// 일반 마법
Skill magic_1 = new MagicBall();
magic_1.GetInfo();
// 파이어 볼
Skill magic_2 = new FireProperty(new MagicBall());
magic_2.GetInfo();
// 일렉트릭 볼
Skill magic_3 = new ElectricProperty(new MagicBall());
magic_3.GetInfo();
// 혼종 : 일렉트릭 파이어 볼
Skill magic_4 = new ElectricProperty(new FireProperty(new MagicBall()));
magic_4.GetInfo();
Console.WriteLine("-----------------------------------\n");
// 일반 실드
Skill shield_1 = new Shield();
shield_1.GetInfo();
// 파이어 실드
Skill shield_2 = new FireProperty(new Shield());
shield_2.GetInfo();
// 일렉트릭 실드
Skill shield_3 = new ElectricProperty(new Shield());
shield_3.GetInfo();
// 혼종 : 일렉트릭 파이어 실드
Skill shield_4 = new ElectricProperty(new FireProperty(new Shield()));
shield_4.GetInfo();
}
}
}
- 실행 결과
<참고 사이트>