[Design Pattern]Singleton
1. 개요
각종 설정 등이 저장된 클래스가 하나 있다고 가정하겠습니다.
프로그램 내에서 여기저기서 마구 접근해서 설정을 바꾸기도 하고 값을 가져오기도 합니다. 이런 클래스는 인스턴스를 하나만 가져야 합니다. 하나 만들어서 쓰는 곳마다 인자로 전달해주면 되긴 합니다만, 접근하는 곳이 많다면, 계속 인자로 전달하는 것은 그다지 바람직하지 않습니다. 전역변수처럼 아무곳에서나 이 인스턴스에 접근을 하면 편하겠죠. Singleton 패턴을 이용하면, 하나의 객체를 만들어서 아무데서나 접근할 수 있습니다.
2. 예제
//--------------------- Singleton으로 구현된 클래스 ---------------- package design.pattern.Singleton; public class SingletonCounter { private static SingletonCounter singleton = new SingletonCounter(); private int cnt = 0; private SingletonCounter(){ } public static SingletonCounter getInstance(){ return singleton; } public int getNextInt(){ return ++cnt; } }
//---------------------- 테스트 클래스 --------------------- package design.pattern.Singleton; public class Test { public static void main(String[] args) { Test t = new Test(); t.Amethod(); t.Bmethod(); } public void Amethod(){ SingletonCounter sc = SingletonCounter.getInstance(); System.out.println("Amethod에서 카운터 호출 " + sc.getNextInt() ); } public void Bmethod(){ SingletonCounter sc = SingletonCounter.getInstance(); System.out.println("Bmethod에서 카운터 호출 " + sc.getNextInt() ); } }
// ---------------------- 실행 결과 -----------------------
// Amethod에서 카운터 호출 1
// Bmethod에서 카운터 호출 2
3. Singleton을 구현하는 몇 가지 방법
//------------- 첫번째 --------------
package design.pattern.Singleton;
public class Singleton1 {
private static Singleton1 single = new Singleton1();
public static Singleton1 getInstance(){
return single;
}
private Singleton1(){
}
}
클래스 로드시 new가 실행이 됩니다. 항상 1개의 인스턴스를 가지게 되겠죠. 코드가 가장 짧고 쉽습니다. 성능도 다른 방법에 비해 좋습니다.
//-------------- 두번째 --------------
package design.pattern.Singleton;
public class Singleton2 {
private static Singleton2 single;
public static synchronized Singleton2 getInstance(){
if (single == null) {
single = new Singleton2();
}
return single;
}
private Singleton2(){
}
}
클래스 로드시에는 인스턴스가 생성되지 않습니다. getInstance()가 처음 호출될 때 생성이 됩니다. 그러나 synchornized가 걸려 있어서 성능이 안 좋습니다. 인스턴스를 사용할 필요가 없을 때는 인스턴스가 생성되지 않는다는 점이 첫번째 방벙에 비해 장점입니다.
//--------------- 세번째 ---------------
package design.pattern.Singleton;
public class Singleton3 {
private volatile static Singleton3 single;
public static Singleton3 getInstance(){
if (single == null) {
synchronized(Singleton3.class) {
if (single == null) {
single = new Singleton3();
}
}
}
return single;
}
private Singleton3(){
}
}
//--------------- 네번째 ---------------
package design.pattern.Singleton;
public class Singleton4 {
private Singleton4(){
}
private static class SingletonHolder{
static final Singleton4 single = new Singleton4();
}
public static Singleton4 getInstatnce(){
return SingletonHolder.single;
}
}
네번째 방법은 내부 클래스를 사용하는 방법입니다. 기존의 3가지 방법에서는 Singleton 클래스가 자기 자신의 타입을 가지는 멤버 변수를 가지고 있는데, 네번째의 경우는 내부 클래스가 가지고 있습니다. 내부 클래스가 호출되는 시점에 최초 생성이 되기 때문에, 속도도 빠르고 필요치 않다면 생성하지도 않습니다.
4. Singleton의 특징
Singleton은 당연히 인스턴스가 1개만 생깁니다. 그렇게 사용하기 위해 만든 것 입니다. 또 하나의 규약은 private 생성자 때문에 상속이 안 된다는 점입니다. (상속받은 하위체는 상위체의 생성자를 호출합니다.) 예를 들어 Singleton에서 설정관련된 xml 파일을 수정한다고 가정 하겠습니다. 상속을 받아 다른 객체를 만들어서 파일을 수정하는 시도를 하면 안됩니다. 상속을 받게 되면 "인스턴스 1개"라는 원칙을 깨게 됩니다.
private 생성자는 외부에서의 직접호출을 통한 생성을 막는 것과 상속을 막는 두 가지 기능을 수행합니다. 둘 다 "인스턴스 1개"라는 원칙을 지키는 것 입니다.
Factory 패턴과 사용법이 매우 유사합니다. Singleton은 Factory의 특이 케이스로 볼 수도 있습니다. Factory는 매번 객체를 만들어서 리턴하는 방법이고 Singleton은 한 개만 만들어서 요청이 들어올 때마다 만들어진 객체를 리턴한다는 게 차이점입니다. 또 일반적으로 Factory는 create...과 같은 메쏘드 이름을 사용하고 Singleton은 getInstance라는 메쏘드 이름을 사용합니다.
위에서 말한 세가지 방법 중 첫번째 방법의 경우는 public으로 멤버 변수를 선언하고 외부에서 직접 변수에 접근해서 사용하게 해도 됩니다. (반드시 private이어야할 필요는 없다는 것 입니다.. ) 두번째와 세번째는 초기화가 보장이 안 되어 있지만, 첫번째의 경우는 보장되어있기 때문입니다. 주의할 점은 외부에서 악의적으로 public 멤버 변수는 바꿔치기를 할 수도 있기 때문에 이런 식으로 접근할 때는 final 을 붙여주는 게 좋습니다.(어차피 private 생성자를 가지고 있으니, 외부에서 새로운 객체를 만들어 낼 수는 없지만 null을 대입할 수는 있기 때문에 final이 필요합니다.) 그럼 public static final이 되는군요! 상수 입니다. 하지만 일반적인 상수와는 다릅니다. 일반적인 상수는 Immutable 로 구현이 되어있기 때문입니다. 상수로 많이 쓰는 String, Integer, Boolean 등은 전부 Immutalbe입니다.
물론 이런 접근이 권장사항은 아닙니다. 그냥 가능하긴 하다는 것 입니다.
5. JAVA API에 있는 Singleton
Boolean에 있는 valueOf 들은 전부 Singleton 비슷하게 구현되어 있습니다. 다만 인자를 받기 때문에 멤버 변수로 예제처럼 1개만 가지고 있는 것이 아니라 여러개를 가질 수 있습니다. true라는 값을 가지는 Boolean과 false라는 값을 가지는 Boolean 객체 2개가 존재하는 것 입니다..
Collections에 있는 empty... 하는 method도 전부 Singleton입니다.
JDK 안에 있는 Singleton은 대부분 위에서 말한 방법 중 첫번째 방법을 쓰고 있습니다.
클래스 로드시 멤버 변수들을 초기화하는 방법입니다. 그래서 대부분 그 멤버 변수들은 public static final 로 선언되어 있습니다.
'개발 > JAVA' 카테고리의 다른 글
[Design Pattern]Composite (0) | 2018.07.16 |
---|---|
[Design Pattern]Strategy (0) | 2018.07.16 |
[Design Pattern]Template Method (0) | 2018.07.13 |
[Design Pattern]Factory (0) | 2018.07.13 |
[Design Pattern]Adapter (0) | 2018.07.12 |