IT 기술
디자인 패턴 - 생성 패턴
cheons
2022. 7. 21. 08:00
728x90
반응형
소스코드 로직 상에서 객체를 생성할 일이 많습니다. 객체가 필요할 때마다, new 키워드로 객체를 생성하고 가비지 컬렉터가 알아서 처리하도록 두는 게 좋을까요? 사실 그다지 좋은 방법이 아닐 것입니다. 프로그램 실행 중 객체는 단 하나만 있어도 되는 경우가 있을 것이고, 특정 객체로부터 복제하여 사용하는 것이 좋을 수 도 있습니다. 뿐만 아니라, 공장처럼 객체를 생성하는 전용 클래스를 만들어서 객체 생성 코드를 관리하는 방법도 있습니다.
이번 포스팅에서는 디자인 패턴 중 생성 패턴 중 싱글턴, 프로토타입, 팩토리 메서드, 빌더, 추상 팩토리 패턴에 대해 알아보겠습니다.
디자인 패턴 정의
- 프로그래머가 애플리케이션이나 시스템을 디자인할 때, 공통된 문제들을 해결하는데 쓰이는 형식화된 가장 좋은 관행
- Programming Language 별로 디자인 패턴을 구현하는 소스코드 구조가 다를 수 있음
생성 패턴 (Creational Pattern)
인스턴스를 만드는 절차를 추상화하는 패턴
싱글턴 (Singleton)
- 프로그램 내에서 오직 하나의 인스턴스가 생성됨을 보장하는 패턴
- 프로그램 어디에서든 인스턴스를 접근할 수 있음
public class Singleton {
private static Singleton singleton = null;
// 외부에서 직접 생성하지 못하도록 private 선언
private Singleton() {
}
// 오직 1개의 객체만 생성
public static Singleton getInstance() {
if (singleton == null) {
singleton = new Singleton();
}
return singleton;
}
}
@Test
public void TestSingleton()
{
for( int i = 0; i < 5; i++ ){
System.out.println(Singleton.getInstance().toString());
}
}
<Console>
SingleObj@98c449
SingleObj@98c449
SingleObj@98c449
SingleObj@98c449
SingleObj@98c449
프로토타입 (Prototype)
- 이미 존재하는 객체를 이용하여, 새로운 객체를 생성하는 패턴
- 객체 생성 시간이 많이 소요될 때 주로 사용 (외부 시스템으로부터 데이터를 수신하여 객체를 생성하는 경우)
- 주로 java의 clone 메서드를 통해 구현
import java.util.ArrayList;
import java.util.List;
/* Prototype */
public class Prototype implements Cloneable {
private List<String> datalist;
public Prototype() {
datalist = new ArrayList<String>();
}
public Prototype(List<String> data) {
this.datalist = data;
}
public List<String> getDataList() {
return this.datalist;
}
public boolean loadData() {
boolean res = true;
// data loading하는 경우는 DB 또는 Http통신 등 다양한 방법으로 이루어 질 수 있음
res &= datalist.add("data1");
res &= datalist.add("data2");
res &= datalist.add("data3");
res &= datalist.add("data4");
return res;
}
@Override
protected Object clone() throws CloneNotSupportedException {
List<String> tmp = new ArrayList<String>(); // 깊은 복사를 통해 객체 내부 데이터를 복제함
for (String data : datalist) {
tmp.add(data);
}
return new Prototype(tmp);
}
}
@Test
public void TestProtoType() throws CloneNotSupportedException {
Prototype proto = new Prototype();
proto.loadData();
Prototype cloneProto = (Prototype)proto.clone();
cloneProto.getDataList().remove(0);
System.out.println("Address : "+proto+"\tOriginal Object : "+proto.getDataList());
System.out.println("Address : "+cloneProto+"\tClone Object : "+cloneProto.getDataList());
}
<Console>
Address : cleancode.designpattern.Prototype@3b7951 Original Object : [data1, data2, data3, data4]
Address : cleancode.designpattern.Prototype@514713 Clone Object : [data2, data3, data4]
팩토리 메서드 (Factory Methods)
- 객체 생성을 서브클래스에게 위임하는 디자인 패턴
- 클래스 생성과 비즈니스 처리 로직을 분리하여 결합도를 낮추기 위해 사용
/* Creator */
public abstract class Factory {
public Shape getShape(ShapeType type) {
return this.create(type);
}
protected abstract Shape create(ShapeType type);
}
/* Creator1 */
public class FactoryShape extends Factory {
@Override
public Shape create(ShapeType type) {
switch (type) {
case Triangle:
return new ShapeTriangle();
case Rectangle:
return new ShapeRectangle();
default:
throw new RuntimeException(type.toString() + "is not existed");
}
}
}
/* Product */
public abstract class Shape {
abstract public String draw();
}
/* Product1 */
public class ShapeRectangle extends Shape{
@Override
public String draw() {
return "Rectangle";
}
}
public class ShapeTriangle extends Shape {
@Override
public String draw() {
return "Triangle";
}
}
public enum ShapeType {
Triangle, Rectangle
}
@Test
public void TestFactoryMethod(){
Factory factory = new FactoryShape();
Shape shapeObj = factory.getShape(ShapeType.Rectangle);
Shape shapeObj2 = factory.getShape(ShapeType.Triangle);
System.out.println(shapeObj.draw());
System.out.println(shapeObj2.draw());
}
<Console>
Rectangle
Triangle
빌더 (Builder)
- 다양한 상태의 객체 생성이 필요할 때 사용하는 객체 생성 패턴
- 객체 생성과 표현부를 분리할 수 있음
/* Product */
public class Pizza {
private String dough = "";
private String sauce = "";
private String topping = "";
public void setDough(String dough) {
this.dough = dough;
}
public void setSauce(String sauce) {
this.sauce = sauce;
}
public void setTopping(String topping) {
this.topping = topping;
}
public String getDough() {
return this.dough;
}
public String getSauce() {
return this.sauce;
}
public String getTopping() {
return this.topping;
}
}
/* Builder */
public abstract class PizzaBuilder {
protected Pizza pizza;
public Pizza getPizza(){
return pizza;
}
public void createNewPizzaProduct(){
pizza = new Pizza();
}
public abstract void buildDough();
public abstract void buildSauce();
public abstract void buildTopping();
}
/* ConcreteBuilder */
public class HawaiianPizzaBuilder extends PizzaBuilder {
@Override
public void buildDough() {
pizza.setDough("cross");
}
@Override
public void buildSauce() {
pizza.setSauce("mild");
}
@Override
public void buildTopping() {
pizza.setTopping("ham,pineApple");
}
}
public class SpicyPizzaBuilder extends PizzaBuilder {
@Override
public void buildDough() {
pizza.setDough("pan baked");
}
@Override
public void buildSauce() {
pizza.setSauce("hot");
}
@Override
public void buildTopping() {
pizza.setTopping("pepperoni,salami");
}
}
/* Director */
public class Cook {
private PizzaBuilder pizzaBuilder;
public void setPizzaBuilder(PizzaBuilder pizzaBuilder) {
this.pizzaBuilder = pizzaBuilder;
}
public Pizza getPizza() {
return pizzaBuilder.pizza;
}
public void constructPizza() {
pizzaBuilder.createNewPizzaProduct();
pizzaBuilder.buildDough();
pizzaBuilder.buildSauce();
pizzaBuilder.buildTopping();
}
}
@Test
public void TestBuilder() {
Cook cook = new Cook();
PizzaBuilder hawaiianPizzaBuilder = new HawaiianPizzaBuilder();
PizzaBuilder spicyPizzaBuilder = new SpicyPizzaBuilder();
cook.setPizzaBuilder(hawaiianPizzaBuilder);
cook.constructPizza();
Pizza pizza = cook.getPizza();
System.out.println(pizza.toString() + "," + pizza.getDough() + "," + pizza.getSauce() + "," + pizza.getTopping());
cook.setPizzaBuilder(spicyPizzaBuilder);
cook.constructPizza();
pizza = cook.getPizza();
System.out.println(pizza.toString() + "," + pizza.getDough() + ","+ pizza.getSauce() + "," + pizza.getTopping());
}
Console
cleancode.designpattern.builder.Pizza@1663380,cross,mild,ham,pineApple
cleancode.designpattern.builder.Pizza@137e0d2,pan baked,hot,pepperoni,salami
추상 팩토리 (Abstract Factory)
- 서로 관련된 객체들을 동시에 생성할 때 이를 추상화된 인터페이스로 객체 생성 방식을 제공하는 패턴
- 팩토리 메서드로 객체를 생성할 경우, 클래스 간 결합도가 증가하는 문제가 발생하여 이를 해결하고자 사용
/* AbstractFactory */
public abstract class GUIFactory {
public abstract Button createButton();
public abstract Scroll createScroll();
}
/* Factory1 */
public class GUILinuxFactory extends GUIFactory {
@Override
public Button createButton() {
return new LinuxButton();
}
@Override
public Scroll createScroll() {
return new VerticalScroll();
}
}
/* Factory2 */
public class GUIWinFactory extends GUIFactory {
@Override
public Button createButton() {
return new WinButton();
}
@Override
public Scroll createScroll() {
return new HorizontalScroll();
}
}
/* Product1 */
public abstract class Button {
public abstract void paint();
}
public class LinuxButton extends Button{
@Override
public void paint() {
System.out.println("Draw Linux Button");
}
}
public class WinButton extends Button{
@Override
public void paint() {
System.out.println("Draw Win Button");
}
}
/* Product2 */
public abstract class Scroll {
public abstract void paint();
}
public class VerticalScroll extends Scroll{
@Override
public void paint() {
System.out.println("VerticalScroll");
}
}
public class WinButton extends Button{
@Override
public void paint() {
System.out.println("Draw Win Button");
}
}
@Test
public void TestAbstractFactory() throws Exception
{
GUIFactory factory = new GUIWinFactory();
Button button = factory.createButton();
Scroll scroll = factory.createScroll();
button.paint();
scroll.paint();
factory = new GUILinuxFactory();
button = factory.createButton();
scroll = factory.createScroll();
button.paint();
scroll.paint();
}
<Console>
Draw Win Button
Draw HorizontalScroll
Draw Linux Button
Draw VerticalScroll
728x90
반응형