public static Boolean valueOf(boolean b) {
return b ? Boolean.TRUE : Boolean.FALSE;
}
장점 :
1 ) 이름을 가질 수 있음
2 ) 호출될때마다 인스턴스를 새로 생성할 필요 없음
3 ) 반환 타입의 하위 타입객체를 반환할 수 있는 능력이 있음
4 ) 입력 매개변수에 다라 매번 다른 클래승의 객체 반환 가능
5 ) 정적 팩토리 메서드를 작성하는 시점에서는 반환할 객체의 클래스가 존재 하지 않아도 됨
단점 :
1) 하위클래스를 만들 수 없다
2) 프로그래머가 찾기 어렵다.
예시) jdbc
: 클래스 안에서 다른 객체를 생성하는 로직이 있다.
// 드라이버 가져와서 디비 생성, 연결하는 아이
public static Connection getConnection() {
try {
//어떤 DB 를 사용할 지 선택, 연결하려는 드라이버 명
Class.forName("org.h2.Driver");
//내가 사용하고자 하는 DB(URL)에 ID/PW 로 접근
return DriverManager.getConnection("jdbc:h2:tcp://localhost/~/testsq",
"sa","");
} catch ( Exception e) {
e.printStackTrace();
}
return null;
}
public class Applepie {
private final int flour;
public final int apple;
public final int suger;
public final int powder;
public static class Builder {
// 필수 매개변수
private final int flour;
public final int apple;
// 선택 매개변수 : 기본값 초기화
public final int suger = 1;
public final int powder = 3;
// 필수 매개변수는 생성자를 사용해 초기화
Builder( int flour, int apple ) {
this.flour = 5;
this.apple = 3
}
// 선택 매개변수는 setter 메서드를 사용함
Builder amount0fsuger(int suger) {
this.suger =suger;
return this;
}
Builder amount0fpowder( int powder) {
this.powder;
return this;
}
}
}
방1) public static final 사용, 생성자는 private
public class Elvis {
// 딱 한번만 인스턴스 호출하였다.
public static final Elvis INSTANCE = new Elvis();
private Elvis() { }
public void mynameis() { }
}
AccessibleObject.setAccessible
을 사용해 private 생성자를 호출하는 공격을 할 수 있다. 이런 공격은 두번째 객체를 생성하려고 할때 예외를 던지게 한다.방2) private static final, 생성자는 private, public 으로 인스턴스 호출 가능
public class Elvis {
// 변수가 private 라 외부에서 호출 불가
private static final Elvis INSTANCE= new Elvis();
private Elvis() { ..}
// public으로 된 인스턴스 호출 메서드가 생김
public static Elvis getInstance() { return INSTANCE; }
public void mynameis() { }
}
방3) enum 으로 원소가 하나뿐인 열거 타입의 싱글턴
public enum Elvis {
INSTANCE;
public void mynameis() { .. }
}
대부분의 상황에서 원소가 하나뿐인 열거 타입이 싱글턴을 만드는게 가장 좋은 방법
private UtilityClass() {
throw new AssertionError();
}
의존성 주입
public class daliycheck() {
private final Todolist todo;
public daliycheck( Todolist todo) {
this.todo = Objects.requireNonNull(todo);
}
}
String 클래스에 있는 matchs는 정규 표현식을 위반하는 문자열을 걸러낼수 있는 좋은 객체이지만,
한번 쓰고 버려지는 가비지 콜랙터의 주 대상이기도 한다. 아래와 같이 미리 캐싱
해두고 사용하는 방법을 사용하면 성능 개선에 도움이 된다. 함수 매번 호출하는 것보다 미리 가져오는게 성능에 좋음.
public class RomanNumerals{
private static final Pattern ROMAN =
Pattern.compile("^(?=[MDCLXVI])M*D?C{0,4}L?X{0,4}V?I{0,4}$");
public static boolean isRomanNumeral(String s){
return ROMAN.matcher(s).matches();
}
}
public Object pop() {
if(size == 0) {
throw new EmptyStackException() ;
}
Object res = elements[--size]'
element[size] = null;
return res;
}
try-finally는 많이 사용하고 익숙하지만 try-with-resources 는 무엇인가 싶다.
try-with-resources 는 일반적으로 복수의 자원을 처리하는 로직이다.
why?
: try-finally 를 사용하면 예외가 try 블록, finally 블록 두개 다 발생가능한다. 이런 경우 추적이 블가능하다.
static void copy(String src, String dst) throws IOException{
try(InputStream in = new FileInputStream(src);
OutputStream out = new FileOutputStream(dst)) {
byte[] buf = new byte[BUFFER_SIZE];
int n;
while((n = in.read(buf)) >= 0){
out.write(buf, 0, n);
}
} catch (IOException e) {
return defaultval;
}
}