Client가 클래스의 인스턴스를 얻는 전통적인 수단은 public 생성자이다.
하지만, 이외에도 클래스는 생성자와 별도로 정적 팩토리 메서드를 제공할 수 있다.
정적 팩토리 메소드
한 Class의 인스턴스를 반환하는 단순한 정적 메소드
ex) Boolean.java
public static Boolean valueOf(boolean b) {
return b ? Boolean.TRUE : Boolean.FALSE
}
(1) 이름을 가질 수 있다
BigInteager(int, int, Random)
vs
BigInteager.probalblePrime(int, Random)
(2) 호출될 때마다 인스턴스를 새로 생성하지는 않아도 된다
(3) 반환 타입의 하위 타입 객체를 반환할 수있는 능력이 있다.
public abstract class Fruit {
abstract void getName();
static public Fruit getNewInstance(String code) {
switch(code) {
case "apple":
return new Apple()
case "banana":
return new Banana()
default: /*some logic*/
}
}
}
class Apple extends Fruit {
public void getName() {
System.out.println("Apple");
}
}
class Banana extends Fruit {
public void getName() {
System.out.println("Banana");
}
}
(4) 입력 매개변수에 따라 매번 다른 클래스의 객체를 반환할 수 있다.
public abstract class Fruit {
protected int property;
static public Fruit getNewInstance() {
if (property == 0) {
return new HotFruit();
}
return new ColdFruit();
}
}
(5) 정적 팩토리 메서드를 작성하는 시점에는 반환할 객체의 클래스가 존재하지 않아도 된다.
public abstract class Fruit {
public abstract void getName();
public static Fruit getNewInstance() {
try {
Class<?> class = Class.forName("com.android.example.Fruit");
return (Fruit) class.newInstance();
} catch (Exception e) {
/* error */
}
}
}
(1) 상속을 하려면, public이나 protected 생성자가 필요하니 정적 팩토리 메서드만 제공하면 하위 클래스를 만들 수 없다.
(2) 정적 팩토리 메서드는 프로그래머가 찾기 어렵다.
큰 생각없이 public 생성자를 이용하여 객체를 생성하지 말고
정적 팩토리 메소드를 사용함으로써 얻는 이득을 알았으니 객체 쓰임에 따라 한번 더 고려하자.