study/Effective Java

[Effective Java] item 3. 생성자나 열거 타입으로 싱글턴임을 보증하라

wookcode 2022. 7. 9. 19:07
728x90
반응형

싱글턴이란?

싱글턴이란 인스턴스를 오직 하나만 생성할 수 있는 클래스를 말한다.

그런데 클래스를 싱글턴으로 만들면 이를 사용하는 클라이언트를 테스트하기가 어려워질 수 있다. 왜냐하면 타입을 인터페이스로 정의한 다음 그 인터페이스를 구현해서 만든 싱글턴이 아니라면 싱글턴 인스턴스를 가짜(mock) 구현으로 대체할 수 없기 때문이다.

다음으로 싱글턴임을 보증하는 3가지 방법을 알아보겠다.


1. private 생성자 + public static final 필드

public class Elvis {
  public static final Elvis INSTANCE = new Elvis();
  private Elvis() { ... }

  public void leaveTheBuilding() { ... }
}

private 생성자는 public static final 필드인 Elvis.INSTANCE를 초기화할 때 딱 한번만 호출된다.

하지만 리플렉션을 통해 private 생성자를 호출하여 싱글턴이 깨질 수 있으므로 flag를 걸어 두번째 객체가 생성되려 할때 예외를 던지게 하면된다.

public class Elvis {
  public static final Elvis INSTANCE = new Elvis();
  private static boolean flag;
  private Elvis() {
      if (flag) {
      throw new ...
    }
    flag = true;
  }

  public void leaveTheBuilding() { ... }
}

2. private 생성자 + 정적팩터리 메서드 + private static final 필드

public class Elvis {
  public static final Elvis INSTANCE = new Elvis();
  private Elvis() { ... }
  public static Elvis getInstance() { return INSTANCE; }

  public void leaveTheBuilding() { ... }
}

Elvis.getInstance 는 항상 같은 객체의 참조를 반환하므로 제2의 인스턴스란 결코 만들어지지 않는다. (역시 리플렉션을 통한 예외는 똑같이 적용된다.)

정적 팩터리 메서드 장점

  • 클라이언트 코드를 변경하지 않고 팩터리 메서드를 변경하여 싱글톤이 아니게 변경할 수 있다.

  • 정적 팩토리를 제네릭 싱글턴 팩토리로 만들 수 있다는 점이다.

    타입은 다르게 싱글톤 객체로 만들 수 있다.

  • 정적 팩터리의 메서드 참조를 공급자로 사용할 수 있다는 점이다.

그러나 위에 두 방식은 직렬화 역직렬화를 거치게 되면 싱글턴이 깨지게 되는 문제가 있다.

그래서 직렬화를 할 때 단순히 Serializable을 구현한다고 선언하는 것만으로는 부족하다.

readResolve 라는 메서드 이름으로 만들게 되면 역직렬화 과정에서 이 메서드를 호출하게 설계가 되어있어 이 메서드까지 선언해줘야 직렬화 역직렬화 과정에서도 싱글턴을 유지할 수 있다.

private Object readResolve() {
    return INSTANCE;
}

3. 원소가 하나인 열거타입

public enum Elvis {
    INSTANSE;

  public void leaveTheBulding() { ... }
}

직렬화도 쉽게 가능하고, 리플렉션 공격에서도 제2의 인스턴스가 생기는 일을 완벽히 막아준다.

3가지 방법중 싱글턴을 만드는 가장 좋은방법이다.

반응형