티스토리 뷰
반응형
Private 생성자나 열거 타입으로 싱글턴임을 보증하라
싱글턴이란,
-
인스턴스를 오직 하나만 생성할 수 있는 클래스를 말한다.
-
싱글턴을 만드는 방식 두 가지
- 두 가지 모두 생성자는 private으로 감춰두고, 유일한 인스턴스에 접근할 수 있는 수단으로 public static 멤버를 하나 마련해둔다.
-
첫번째 방식
//public static final 필드 방식의 싱글턴 public class Elvis { public static final Elvis INSTANCE = new Elvis(); private Elvis(){...}; }
- private 생성자는 public static final 필드인 Elvis.INSTANCE를 초기화할 때 딱 한번만 호출된다.
- public 이나 protected(같은 패키지 혹은 상속받을 시 접근 가능) 생성자가 없으므로 Elvis 클래스가 초기화될 때 만들어진 인스턴스가 전체 시스템에서 하나뿐임을 보장한다. 이는, 클라이언트가 손 쓸 방법이 없다는 얘기다.
- but, 예외가 있는데.. 이는 권한이 있는 클라이언트는 리플렉션 API(아이템 65)인 AccessibleObject.setAccessible을 사용해 private생성자를 호출할 수 있다.
- 예외를 방어하려면, 생성자를 수정하여 두 번째 객체가 생성되려할 때 예외를 던지게 하면 된다.
- 장점
- 해당 클래스가 싱글턴임이 API에 명백히 드러나 있다.
- 간결함이다.
-
두번째 방식
//정적 팩토리 방식의 싱글턴 public class Elvis { private static final Elvis INSTANCE = new Elvis(); private Elvis(){...} public static Elvis getInstance() { return INSTANCE; } }
- Elvis.getInstacne는 항상 같은 객체의 참조를 반환하므로 제2의 Elvis 인스턴스란 결코 만들어지지 않다(이 또한, 리플렉션을 통한 예외는 똑같이 적용됨)
- 장점
- API를 바꾸지 않고도 싱글턴이 아니게 변경할 수 있다.
- 원한다면 정적 팩토리를 제네릭 싱글턴 팩토리로 만들 수 있다.(아이템 30)
- 싱글턴 부분을 팩토리로 만들어 팩토리 접근시에만 객체 생성할 수 있도록 하겠다.
- 정적 팩토리의 메서드 참조를 공급자(supplier)로 사용할 수 있다.
- Elvis::getInstance를 Supplier로 사용하는 식이다(아이템 43, 44)
- 리플렉션이란,
- 구체적인 클래스 타입을 알지 못해도 컴파일된 바이트 코드를 통해 해당 클래스의 메소드, 타입, 변수까지 접근가능한 자바 API를 말한다.
- 즉, 리플렉션은 컴파일된 바이트 코드에 접근이 가능한 것이다.(static 영역 접근 가능)
- 그로 인해, 런타임 후 생성되는 인스턴스들의 멤버 변수가 private으로 선언되어 있어도, 클래스 이름만 알고 있다면 클래스의 정보를 가져 올 수 있다. 결과적으로 private으로 생성자를 선언해도 리플렉션으로 인해 싱글톤이 깨질 수 있다는 말이다.
-
세번째 방법
-
private이 아닌 enum 타입 방식의 싱글턴을 말한다. 얘가 가장 바람직한 방법이다.
- enum은 인스턴스 생성과 상속을 방지하고 상수값의 타입 안정성을 보장한다.
- 그로인해, 인스턴스를 생성할 시 multi thread로 부터 안전하고, 직렬화가 자동으로 처리 된다.
- 결과적으로, 리플렉션으로 인한 싱글톤 깨짐 현상은 발생하지 않는다. 왜? enum 타입은 static 영역에 올라가고 타입 안정성을 보장해주니깐!
-
public enum Elvis { INSTANCE; } //사용 Single single = Elvis.INSTANCE;
-
-
대부분 상황에서는 원소가 하나뿐인 열거 타입이 싱글턴을 만드는 가장 좋은 방법이다.
-
만들려는 싱글턴이 Enum 외의 클래스를 상속해야 한다면 이 방법은 사용할 수 없다.(열거 타입이 다른 인터페이스를 구현하도록 선언 할 수는 있다.)
-
리플렉션 대응 가능?
- enum 타입을 사용하게 되면 해당 변수들은 static final 변수가 되므로 static영역에 올라가게 된다.
- 그로 인해, 리플렉션에서 따로 접근이 불가능하여 대응이 가능하다.
http://www.yes24.com/Product/Goods/65551284?scode=032&OzSrank=1
반응형
'Java > Effective Java' 카테고리의 다른 글
[Effective Java] 아이템6. 자원을 직접 명시하지 말고 의존 객체 주입을 사용하라 (0) | 2019.12.22 |
---|---|
[Effective Java] 아이템5. 불필요한 객체 생성을 피하라 (0) | 2019.12.22 |
[Effective Java] 아이템 4. 인스턴스화를 막으려거든 private 생성자를 사용하라 (0) | 2019.12.14 |
[Effective Java] 아이템2. 생성자에 매개변수가 많다면 빌더를 고려하라 (0) | 2019.12.14 |
[Effective Java] 아이템1. 생성자 대신 정적 팩토리 메서드를 고려하라 (0) | 2019.12.14 |
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
- Total
- Today
- Yesterday
링크
TAG
- 김영한
- ifPresent
- flatMap
- Effective Java
- effectivejava
- java8
- jdk버전
- 점층적 생성 패턴
- 스프링부트
- 복사 팩토리
- package-private
- 이펙티브 자바
- 빌더 패턴
- java
- try with resources
- 인프런
- 빈 순환 참조
- @Lazy
- 생성자
- JPA
- mustache
- try catch finally
- 연관관계
- springboot
- 자바8
- 팩토리 메소드 패턴
- junit
- Spring
- 이펙티브자바
- 정적팩터리메서드
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
글 보관함