Design pattern(creational 생성 패턴) with Java
27 Apr 2022생성 패턴(creational patterns)
- 시스템이 사용하는 구체적인 클래스에 대한 지식을 캡슐화
- 이러한 구체적인 클래스의 인스턴스가 생성되고 결합되는 방법을 숨기는 것입니다.
생성 디자인 패턴은 객체 생성 패턴과 클래스 생성 패턴으로 더 분류되며, 여기서 객체 생성 패턴은 객체 생성을 처리하고 클래스 생성 패턴은 클래스 인스턴스화를 처리합니다.
더 자세히 말하면 객체 생성 패턴은 객체 생성의 일부를 다른 객체로 위임하는 반면 클래스 생성 패턴은 객체 생성을 하위 클래스로 위임합니다.
- 싱글톤 패턴 클래스가 하나의 인스턴스만 갖도록 하고 이에 대한 전역 액세스 지점을 제공하는 패턴
- 팩토리 메소드 패턴 클래스가 인스턴스화를 서브클래스로 위임할 수 있도록 하는 패턴
- 추상 팩토리 패턴 객체의 구체적인 클래스를 지정하지 않고 관련 또는 종속 객체를 생성하기 위한 인터페이스를 제공하는 패턴
- 빌더 패턴 동일한 생성 프로세스가 다른 표현을 생성할 수 있도록 복잡한 객체의 생성을 표현과 분리하는 패턴
- 프로토타입 패턴 프로토 타입 인스턴스를 사용하여 생성할 객체의 종류를 지정하고 이 프로토타입을 복제하여 새 객체를 생성하는 패턴
싱글톤 패턴(Singleton)
인스턴스를 하나만 생성해야할 경우 사용하는 패턴
시스템 런타임, 환경 세팅에 대한 정보 등, 인스턴스가 여러개 일 때 문제가 생길 수 있는 경우가 있다.
인스턴스를 오직 한개만 만들어 제공하는 클래스가 필요하다
1. private 생성자에 static 메소드
-> 멀티쓰레드 환경에서 안전하지 않은 단점 존재
public class SingletonBasic {
SingletonBasic instance;
private SingletonBasic() {}
public static SingletonBasic getInstance() {
if (instance == null) {
instance = new SingletonBasic();
}
return instance;
}
}
2. 동기화(synchronized)를 사용해 멀티쓰레드 환경에 안전하게 만드는 방법
-> 장점: 하나의 쓰레드만 허용하므로 1번의 문제는 해결 되었으나
-> 단점: 메서드 호출 시마다 동기화 락이 걸려 약간의 부하가 생기는 것이 단점
public class SingletonSync {
private static SingletonSync instance;
private SingletonSync() {}
public static synchronized SingletonSync getInstance() {
if (instance == null) {
instance = new SingletonSync();
}
return instance;
}
}
3. 이른 초기화 (eager initialization)을 사용하는 방법
-> 장점: SingletonEager class가 로딩되는 시점에 인스턴스가 만들어지기 때문에 thread safe 함.
-> 단점: 이 인스턴스를 만드는데 메모리도 많이 사용하고 오래 걸리는 작업인데 쓰지 않는다면 불필요한 작업이 됨.
public class SingletonEager {
private static final SingletonEager INSTANCE = new SingletonEager();
private SingletonEager() {}
public static SingletonEager getInstance() {
return INSTANCE;
}
}
4. double checked locking으로 효율적인 동기화 블럭 만들기
-> 장점
- Thread safe
- synchronize가 다 걸리지 않아 성능 유리, 필요한 시점에 생성(lazy)
-> 단점
- jdk 1.5 버전부터 가능
- 복잡한 volatile 개념이 들어감.
public class SingletonDoubleCheck {
private static volatile SingletonDoubleCheck instance;
private SingletonDoubleCheck() {}
public static SingletonDoubleCheck getInstance() {
if (instance == null) {
synchronized (SingletonDoubleCheck.class) { // class를 lock으로 쓰기
if (instance == null) {
instance = new SingletonDoubleCheck();
}
}
}
return instance;
}
}
5. static inner 클래스를 사용하는 방법
-> 장점
- Thread safe함.
- getInstance가 호출 될떄 SingletonInnerHolder 클래스가 로딩되고 그때 인스턴스를 만듦
-> 단점: 리플렉션(SingletonInner.class.getDeclaredConstructor())이나 직렬화 과정이 추가 된다면 싱글톤을 강제할 수 없다.
public class SingletonInner {
private SingletonInner() {}
private static class SingletonInnerHolder {
private static final SingletonInner INSTANCE = new SingletonInner();
}
public static SingletonInner getInstance() {
return SingletonInnerHolder.INSTANCE;
}
}
6. enum을 사용하는 방법
-> 장점
- Thread safe함
- 리플랙션에도 안전함
- 기본적으로 Serialize를 구현하고 있음
-> 단점
- 미리 만들어짐(eager)
- 클래스이기 때문에 단일 상속 한계가 있다.
public enum SingletonEnum {
INSTANCE;
}
팩토리 메서드 패턴(Factory method)
어떤 객체(인스턴스)를 생성하는 것을 구체적인 클래스(concrete class)가 아닌 추상적인 인터페이스의 메서드로 감싸는 것
- factory 역할을 할 인터페이스 생성(Creator)
- 인터페이스에 바뀌지 않을 구현부는 작성하고, 바뀌는 부분은 추상메서드로 생성
- default 메서드(java8), private 메서드(java9)로 구성 가능
- 구체적인 클래스(하위 클래스)에서 구체적인 메서드 구현
- factory에서 만든 오브젝트의 타입인 Product를 다양하게 만들 수 있다.
- concreteCreator 와 마찬가지로 ConcreteProduct도 여러개를 갖을 수 있다.
[Creator - ShipFactory]—-> [Product - Ship]
[ConcreteCreator -WhiteshipFactory]
장점
- OCP 원칙을 지켜낸 방식이다.(기존 로직을 수정하지 않고 새로운 객체를 만들 수 있다.)
- 새로운 인스턴스를 다른 방법으로 확장 구현이 가능하다.
- Creator - Product 간에 결합도를 낮아져 유연해졌다.
단점
- 클래스가 늘어나고, 복잡도가 증가함
추상 팩토리(Abstract factory)
팩토리를 추상화된 형태로 제공하는 것(인터페이스 or 추상클래스)
구체적인 팩토리에서 구체적인 인스턴스를 만드는 과정은 팩토리 메서드 패턴과 유사함.
그러나 초점이 팩토리를 사용하는 클라이언트에 맞춰져 있으면 추상 팩토리 메서드 패턴이다.
바꾸는 것 없이 제품군을 늘려가는 방법
팩토리 메서드 vs 추상 팩토리 비교
둘 다 객체를 만드는 과정을 추상화했다.
- 팩토리 메소드 패턴: 팩토리를 구현하는 방법 (inheritance)에 초점
- 추상 팩토리 패턴: 팩토리를 사용하는 방법 (composition)에 초점
팩토리 메서드 패턴
은 Concrete(구체적인) 타입의 인스턴스를 만드는 과정을 구체적인 팩토리(하위클래스)로 옮기는 것이 목적
추상팩토리 패턴
은 관련 있는 여러 객체를 추상화 시켜서 사용할 수 있도록 하는 것(구체적인 타입에 의존하지 않게끔)
클라이언트 관점에서 팩토리를 통해서 추상화된 인터페이스만 쓸 수 있게 하는 것.
구체적인 클래스를 사용할 필요가 없어진다.
빌더 패턴(builder)
동일한 프로세스를 거쳐 다양한 구성의 인스턴스를 만드는 방법
Builder 인터페이스만 사용하여 빌더패턴을 구현할 수 있고, Director를 사용해서 구현할 수 있다.
Director를 사용한다면 중복되는 builder 스택을 Director에 숨겨놓고 클라이언트는 Director를 통해서 product를 받아서 사용한다.
- 빌더 인터페이스 생성
- 필드 체이닝을 위해 빌더 클래스를 리턴하도록 한다.
- 마지막은 Product를 리턴한다.(getProduct - 이곳에서 validation 체크해도 됨)
장점
- 빌더를 사용하지 않는다면 생성자 여러개를 만들거나 null로 세팅하거나 해야한다.
- 순서 상관없이 필요한 구성으로 생성 가능하다.
- 불완전한 객체의 생성을 막을 수 있다.
단점
- 빌더 클래스를 생성 해야 한다는 리소스가 추가된다.(+ Director)
- 기존 대비 구조가 좀 더 복잡해진다.
프로토타입(Prototype) 패턴
기존 인스턴스를 복제하여 새로운 인스턴스를 만드는 방법
- 복제 기능을 갖추고 있는 기존 인스턴스를 프로토타입으로 사용해 새 인스턴스를 만들 수 있다
x.clone() != x;
// will be true, and that the expression:
x.clone().getClass() == x.getClass();
// will be true, but these are not absolute requirements. While it is typically the case that:
x.clone().equals(x);
// will be true, this is not an absolute requirement.
- ArrayList에서 제공하는 복사 :
new ArrayList<>(source);
- ModelMapper에서 제공하는 복사:
modelMapper.map(source, destinationType)
- java가 제공하는 clone 사용
- Cloneable 인터페이스를 구현(protected 접근지시자 때문)
- clone 메서드 오버라이딩
- 필요한 경우 clone 메서드에서 사용중인 다른 객체의 인스턴스 생성
기본적으로 shallow copy
를 지원하기 때문에 복사한 객체에서 사용하는 인스턴스들은 같은 객체를 사용한다.
만약 완벽히 독립된 clone(deep copy)을 원할 경우 오버라이딩한 clone 메서드에서 새로운 인스턴스를 생성해서 사용해야 한다.
장점
- 복잡한 객체를 만드는 과정을 숨길 수 있다.
- 동일한 작업을 반복하지 않아도 된다.
- DB, 네트워크를 사용하는 등 리소스를 많이 사용하는 작업인 경우 복제의 이점이 많다.
- 추상적인 타입을 리턴할 수 있다.
단점
- 동일한 객체를 만들어야하는데 객체간에 순환참조가 있는 경우 직접 구현해야하는 복잡도가 있다.