반응형

1. 서론

이번 포스팅에서는 Chapter13의 디폴트 메서드에 대해 진행하도록 하겠습니다.

 

2. 변화하는 API

인터페이스에 새로운 메서드를 추가하는 등 인터페이스를 바꾸고 싶을때는 문제가 발생합니다. 

 

변경 인터페이스를 상속하고 있는 클래스들에 모두 영향이 가기 때문이죠.

 

이러한 문제점을 java에서는 디폴트 메서드라는 개념을 도입하여  해결하였습니다.

 

디폴트 메서드가 왜 필요한지 예제를 통해 알아보겠습니다.

 

처음 아래처럼 Resizable 인터페이스를 만들어 제공하였습니다.

 

public interface Resizable {
    int getWidth();
    int getHeight();
    void setWidth();
    void setHeight();
    void setAbsoluteSize(int width, int height);
}

public class Ellipse implements Resizable {
	...
}

 

시간이 지나 요구사항을 처리하다보니 Resizable에 setRelativeSize(int wFactor, int hFactor) 가 필요함을 깨달았습니다.

 

이때 바로 문제가 생깁니다.

 

Resizable를 구현한 클래스들은 모두 setRelativeSize 메서드를 구현해야 하기 때문입니다.

 

3. 디폴트 메서드란 무엇인가?

디폴트 메서드는 인터페이스에 구현 메서드를 놓을 수 있는 새로운 시그니쳐입니다.

 

이를 통해, 위와같은 문제점을 해결할 수 있습니다.

 

사용법으로는 default 라는 키워드로 메서드를 정의하면 됩니다.

 

아래는 디폴트 메서드를 사용한 예제입니다.

 

public interface Sized {
    int size();
    default boolean isEmpty() {
        return size() == 0;
    }
}

 

 

 

 

 

 

 

반응형

 

 

 

 

 

 

 

4. 디폴트 메서드 활용 패턴

디폴트 메서드를 활용하는 방식으로는 두가지가 있습니다.

 

  • 선택형 메서드
  • 동작 다중 상속

 

1) 선택형 메서드

 

인터페이스에는 간혹 구현 클래스에서 크게 중요하지 않은 메서드를 정의할때가 있습니다.

 

그로인해, 구현 클래스들은 메서드 오버라이드만 할 뿐 실제 바디에는 내용이 없게 하는 코드들이 생기게 됩니다.

 

하지만 디폴트 메서드를 사용하면 기본 메서드 구현을 인터페이스에 상주할 수 있어 이러한 문제를 해결하게 됩니다.

 

2) 동작 다중 상속

 

디폴트 메서드를 통해 여러 인터페이스를 상속하고 있는 클래스들은 사용할 수 있는 동작이 풍부해집니다.

 

아래는 코드로 해당 이점을 보여주는 예제입니다.

 

public interface Rotatable {
    void setRotationAngle(int angleInDegrees);
    int getRotationAngle();
    default void rotateBy(int angleInDegrees) {
        setRotationAngle(getRotationAngle() + angleInDegrees);
    }
}

public interface Resizable {
    int getWidth();
    int getHeight();
    void setWidth();
    void setHeight();
    void setAbsoluteSize(int width, int height);
    default void setRelativeSize(int wFactor, int hFactor) {
        setAbsoluteSize(getWidth() / wFactor, getHeight() / hFactor);
    }
}

public interface Moveable {
    int getX();
    int getY();
    void setX(int x);
    void setY(int y);
    default void moveHorizontally(int distance) {
        setX(getX() + distance);
    }

    default void moveVertically(int distance) {
        setY(getY() + distance);
    }
}

public class Moster implements Rotatable, Moveable, Resizable {

}

 

5. 해석 규칙

디폴트 메서드로 인해 개발자에게 장점만 제공된것은 아닙니다.

 

단점으로는 같은 시그니처를 갖는 디폴트 메서드를 상속 받는 상황이 생길 수 있다는 것입니다.

 

자바에서는 이러한 경우를 대비해 아래와 같은 3가지 규칙을 세워 메서드의 상속 우선순위를 가지도록 했습니다.

 

  1. 클래스가 항상 이긴다. 클래스나 슈퍼클래스에서 정의한 메서드가 디폴트 메서드보다 우선권을 갖습니다.
  2. 1번 다음으로는 서브 인터페이스가 이깁니다.
  3. 1번과 2번과 같은 상황이 없고, 디폴트 메서드에 대해 순위가 정해지지 않은 경우에는 상속 클래스에서 명시적으로 디폴트 메서드를 오버라이드하여 호출해야 합니다.

 

6. 마무리

이번 포스팅에서는 Chapter13 디폴트 메서드 대해 진행하였습니다.

다음에는 Chapter14 자바 모듈 시스템에 대해 포스팅하겠습니다.

반응형
반응형

1. 서론

 

이번 포스팅에서는 Chapter12의 새로운 날짜와 시간 API에 대해 진행하도록 하겠습니다.

 

2. LocalDate, LocalTime, Instant, Duration, Period 클래스

java.time 패키지에는 LocalDate, LocalTime, LocalDateTime, Instant, Duration, Period 등의 새로운 클래스를 제공합니다.

 

1) LocalDate 와 LocalTime 사용

 

LocalDate 인스턴스는 시간을 제외한 날짜를 표현하는 불변 객체입니다.

 

LocalDate는 팩토리 메서드 of를 통해 생성이 가능합니다.

 

아래는 2017년 9월 21일의 LocalDate를 만들어 사용하는 예제입니다.

 

LocalDate date = LocalDate.of(2017, 9, 21);
int year = date.getYear(); // 2017
Month month = date.getMonth(); // 9월
int day = date.getDayOfMonth(); // 21
DayOfWeek dow = date.getDayOfWeek(); // 목요일 
int len = date.lengthOfMonth(); // 31 (월의 일 수)
boolean deal = date.isLeapYear(); // false (윤년 유무)

 

현재 날짜를 구할때는 팩토리 메서드 now를 사용하면 됩니다.

아래는 예제입니다.

 

LocalDate today = LocalDate.now();

 

아래는 LocalDate 객체에서 값을 얻는 또 다른 방법입니다.

 

int year = date.get(ChronoField.YEAR);
int month = date.get(ChronoField.MONTH_OF_YEAR);
int day = date.get(ChronoField.DAY_OF_MONTH);

 

 

LocalTime은 날짜가 아닌 시간을 제공하는 클래스입니다.

 

LocalTime 역시 of 메서드를 통해 생성이 가능하며, (시, 분), (시, 분, 초) 각각 인수로 받아 생성 가능하도록 오버로드 되어 있습니다.

 

아래는 13시 24분 50초 를 만드는 예제 입니다.

 

LocalTime time = LocalTime.of(13, 45, 20);
int hour = time.getHour();
int minute = time.getMinute();
int second = time.getSecond();

 

날짜 혹은 시간의 정보를 담고있는 문자열에서 LocalDate, LocalTime으로 만들고 싶은 경우가 있습니다.

이를 위해, LocalDate, LocalTime 은 parse 메서드를 제공합니다.

 

아래는 예제 입니다.

 

LocalDate date = LocalDate.parse("2017-09-21");
LocalTime time = LocalTime.parse("13:45:20");

 

2) 날짜와 시간 조합

 

LocalDateTime은 날짜와 시간 정보를 모두 가지고 있는 클래스입니다.

 

아래는 LocalDateTime 을 만드는 예제입니다.

 

LocalDateTime dt1 = LocalDateTime.of(2017, Month.SEPTEMBER, 21, 13, 45, 20);
LocalDateTime dt2 = LocalDateTime.of(date, time);
LocalDateTime dt3 = date.atTime(13 ,45 ,20);
LocalDateTime dt4 = date.atTime(time);
LocalDateTime dt5 = time.atDate(date);

 

반대로 LocalDateTime 에서 LocalDate, LocalTime 만을 빼내고 싶은 경우에는 아래와 같이 toLocalDate, toLocalTime 메서드를 사용하면 됩니다.

 

LocalDate date1 = dt1.toLocalDate();
LocalTime time1 = dt1.toLocalTime();

 

 

3) Instant 클래스 : 기계의 날짜와 시간

 

Instant 클래스는 유닉스 에포크 시간을 기준으로 특정 지점까지의 시간을 초로 표현합니다.

또한, 이 클래스는 나노초의 정밀도를 제공합니다.

 

아래는 Instant 인스턴스를 만드는 예제입니다.

 

Instant.ofEpochSecond(3);
Instant.ofEpochSecond(3, 0);
Instant.ofEpochSecond(2, 1000000000);
Instant.ofEpochSecond(4, -1000000000);

 

Instant 클래스 역시 정적 메서드로 now를 제공합니다.

 

Instant는 초, 나노초 정보를 가지고 있기 때문에 LocalDate, LocalTime와 같이 사람이 읽을 수 있는 정보를 제공하지 않습니다.

 

대신 Period와 Duration와는 함께 활용할 수 있습니다.

 

4) Duration 과 Period 정의

 

Duration은 두 시간 차의 정보를 가지고 있는 클래스입니다.

 

아래는 Duration을 생성하는 예제입니다. 생성시에는 LocalTime, LocalDateTime, Instant를 활용할 수 있습니다.

 

Duration d1 = Duration.between(time1, time2);
Duration d2 = Duration.between(dateTime1, dateTime2);
Duration d3 = Duration.between(instant1, instant2);

 

Period는 시간이 아닌 두 날짜 차의 정보를 가지고 있는 클래스입니다.

 

아래는 Period를 생성하는 예제입니다.

 

Period tenDays = Period.between(LocalDate.of(2017, 9, 11), LocalDate.of(2017, 9, 21));

 

Duration과 Period는 자체적으로도 생성이 가능하도록 팩토리 메서드들을 제공합니다.

 

아래는 예제입니다.

 

Period tenDays = Period.ofDays(10);
Period threeWeeks = Period.ofWeeks(3);
Period twoYearsSixMonthsOneDay = Period.of(2, 6, 1);

 

지금까지 알아본 클래스들은 모두 불변 객체로 생성하여 제공합니다.

 

 

 

반응형

 

 

 

3. 날짜 조정, 파싱, 포매팅

기존 날짜, 시간 데이터를 절대적으로 변경하고 싶은 경우가 있습니다.

 

이런 경우 with 관련 메서드를 통해 가능합니다.

with 메서드를 사용한다고 기존 객체의 값이 변하는것이 아니며, 새로운 객체에 값만을 바꿔 제공합니다.
이는 불변 객체라는 특징을 제공하기 위해서 입니다.

 

아래는 예제입니다.

 

LocalDate date1 = LocalDate.of(2017, 9 , 21);
LocalDate date2 = date1.withYear(2011); // 2011-09-21
LocalDate date3 = date2.withDayOfMonth(25); // 2011-09-25
LocalDate date4 = date3.with(ChronoField.MONTH_OF_YEAR, 2); // 2011-02-25

 

 

상대적인 방법으로도 변경도 가능합니다.

 

아래는 예제입니다.

 

LocalDate date1 = LocalDate.of(2017, 9, 21);
LocalDate date2 = date1.plusWeeks(1);
LocalDate date3 = date2.minusYears(6);
LocalDate date4 = date3.plus(6, ChronoUnit.MONTHS);

 

1) TemporalAdjusters 사용하기

 

복잡한 날짜 조정기능이 필요한 경우 with 메서드에 TemporalAdjusters 를 사용하여 해결할 수 있습니다.

 

아래는 TemporalAdjusters를 사용하는 한 예제입니다.

 

LocalDate date1 = LocalDate.of(2014, 3, 18); // 2014-03-18
LocalDate date2 = date1.with(TemporalAdjusters.nextOrSame(DayOfWeek.SUNDAY)); // 2014-03-23
LocalDate date3 = date2.with(TemporalAdjusters.lastDayOfMonth()); // 2014-03-31

 

TemporalAdjusters 에서 제공하는 메서드가 없는 경우에는 TemporalAdjuster 함수형 인터페이스를 구현하여 사용하면 됩니다.

 

아래는 TemporalAdjuster 입니다.

 

@FunctionalInterface
public interface TemporalAdjuster {
    Temporal adjustInto(Temporal temporal);
}

 

 

2) 날짜와 시간 객체 출력과 파싱

 

날짜와 시간 관련에서는 포매팅과 파싱은 서로 떨어질 수 없는 관계입니다.

 

때문에, java에서는 DateTimeFormatter를 제공하여 손쉽게 날짜나 시간을 특정 형식의 문자열로 만들어 줍니다.

 

아래는 예제입니다.

 

LocalDate date = LocalDate.of(2014, 3, 18);
String s1 = date.format(DateTimeFormatter.BASIC_ISO_DATE);
String s2 = date.format(DateTimeFormatter.ISO_LOCAL_DATE);

 

반대로 문자열에서 날짜나 시간으로도 변환시 사용할 수도 있습니다.

 

아래는 예제입니다.

 

LocalDate date1 = LocalDate.parse("20140318", DateTimeFormatter.BASIC_ISO_DATE);
LocalDate date2 = LocalDate.parse("2014-03-18", DateTimeFormatter.ISO_LOCAL_DATE);

 

DateTimeFormatter 은 기존의 java.util.DateFormat과 달리 쓰레드에 안전합니다.

 

추가로 DateTimeFormatter 은 특정 패턴으로도 포매팅을 제공합니다.

 

DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd/MM/yyyy");
LocalDate date1 = LocalDate.of(2014, 3, 18);
String formattedDate = date1.format(formatter);
LocalDate date2 = LocalDate.parse(formattedDate, formatter);

 

좀 더 복합적인 포매팅을 원할시에는 DateTimeFormatterBuilder 를 사용하면 됩니다.

 

4. 마무리

이번 포스팅에서는 Chapter12 새로운 날짜와 시간 API에 대해 진행하였습니다.

다음에는 Chapter13 디폴트 메서드에 대해 포스팅하겠습니다.

반응형
반응형

1. 서론

이번 포스팅에서는 Chapter11의 null 대신 Optional 클래스에 대해 진행하도록 하겠습니다.

 

2. 값이 없는 상황을 어떻게 처리할까?

책에서는 null 관련하여 아래 예제를 말합니다.

 

public class Person {
    private Car car;

    public Car getCar() {
        return car;
    }
}
    
public class Car {
    private Insurance insurance;

    public Insurance getInsurance() {
        return insurance;
    }
}
    
public class Insurance {
    private String name;
    
    public String getName() {
        return name;
    }
}

public String getCarInsuranceName(Person person) {
    return person.getCar().getInsurance().getName();
}

 

위의 코드는 Person이 차를 가지고 있지 않는 경우에는 NullPointerException 이 발생하게 됩니다.

 

1) 보수적인 자세로 NullPointerException 줄이기

 

위 예제에서 null를 예방하기 위해 if-else 를 사용하게 된다면 아래와 같은 코드가 될 것입니다.

 

public String getCarInsuranceName(Person person) {
    if (person != null) {
        Car car = person.getCar();
        if (car != null) {
            Insurance insurance = car.getInsurance();
            if (insurance != null) {
                return insurance.getName();
            }
        }
    }
    return "Unknown";
}

 

한눈에 봐도 코드가 난잡하며, 객체의 연관도가 깊을수록 if의 깊이는 증가됩니다.

 

가끔 깊이가 너무 깊어져 이를 예방하기 위해 아래와 같은 코드가 나올수도 있습니다.

 

public String getCarInsuranceName(Person person) {
    if (person == null) {
        return "Unknown";
    }
    Car car = person.getCar();
    if (car == null) {
        return "Unknown";
    }
    Insurance insurance = car.getInsurance();
    if (insurance == null) {
        return "Unknown"; 
    }
    
    return insurance.getName();
}

 

 

이 코드의 경우에는 깊이는 깊어지지 않지만 너무 많은 return 출구가 있어 유지보수가 어렵다는 단점을 가지게 됩니다.

 

 

 

2) null 때문에 발생하는 문제

 

위와 같이 null로 인해서 발생하는 문제로는 아래와 같습니다.

 

  1. 에러의 근원 : NullPointerException 은 자바에서 가장 흔히 발생하는 에러입니다.
  2. 코드를 어지럽힘 : null 체크를 통해 코드가 난잡하고 어지럽게 됩니다.
  3. 아무 의미가 없음 : null은 아무 의미도 표현하지 않으며, 이는 값이 없음을 표현하기에 부적합 합니다.
  4. 자바 철학에 위배된다 : 자바의 경우, 개발자에게 모든 포인터를 숨겼지만 null은 유일하게 포인터를 숨길 수 없었습니다.
  5. 형식 시스템에 구멍을 만든다 : null은 무형식이며 정보를 포함하고 있지 않으므로 모든 참조 형식에 null을 할당 할 수 있고, 이는 결국 위험한 코드를 만들게 됩니다.

 

 

 

 

 

 

 

반응형

 

 

 

 

 

 

 

3. Optional 클래스 소개

 

자바 8 에서는 null 문제를 해결하기 위해 Optional<T> 라는 클래스를 제공합니다.

 

이는 사실상 단순히 null를 위한 Wrapper 클래스입니다.

 

아래는 Optional 를 간단히 보여주는 그림입니다.

 

 

 

모든 null 참조를 Optional로 대치하는것은 바람직하지 않습니다.

Optional은 객체에 대해서 null 체크, null 인 경우 대처를 어떻게 할지를 도와주는 역할입니다.

 

때문에, Optional를 사용한다고 NullPointerException이 나지 않는것은 아닙니다.

 

4. Optional 적용 패턴

 

1) Optional 객체 만들기

 

Optional 도 클래스이기 때문에 객체를 생성해서 사용해야 합니다.

 

1. 빈 Optional

 

아래와 같이 빈 Optional 객체를 생성할 수 있습니다.

 

Optional<Car> optCar = Optional.empty();

 

2. null이 아닌 값으로 Optional 만들기

 

아래와 같은 기존에 있는 객체로 Optional 객체를 만들 수도 있습니다.

 

Optional<Car> optCar = Optional.of(car);

 

단, 이 경우 인자인 car가 null인 경우 NullPointerException이 발생합니다.

 

 

3. null 값으로 Optional 만들기

 

아래와 같은 방법으로도 Optional 객체를 만들 수 있습니다.

 

Optional<Car> optCar = Optional.ofNullable(car);

 

2번의 of 메서드와의 차이점으로는 car가 null인 경우 NullPointerException 가 아닌 빈 Optional을 반환한다는 점입니다.

 

 

2) 맵으로 Optional의 값을 추출하고 변환하기

 

Optional 클래스는 스트림 메서드와 비슷한 map 메서드를 지원합니다.

 

아래와 같이 map 을 사용할 수 있습니다.

 

Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
Optional<String> name = optInsurance.map(Insurance::getName);

 

만약, Optional이 비어있으면 아무일도 일어나지 않습니다.

 

 

3) flatMap으로 Optional 객체 연결

 

스트림 메서드의 flatMap 비슷한 기능의 메서드도 제공하고 있습니다.

 

Optional의 flatMap도 스트림 메서드와 동일하게 Optional<Optional<T>> 와 같은 depth 가 생기는 것을 Optional<T> 로 평준화 해주는 메서드입니다.

 

 

4) Optional로 자동차의 보험회사 이름 찾기

 

그럼 이제 배운 Optional를 사용하여 예제를 해결하게 되면 아래와 같은 코드가 나오게 됩니다.

 

public class Person {
    private Car car;
    public Optional<Car> getCarAsOptional() {
        return Optional.ofNullable(car);
    }
}

public class Car {
    private Insurance insurance;
    public Optional<Insurance> getInsuranceAsOptional() {
        return Optional.ofNullable(insurance);
    }
}

public class Insurance {
    private String name;
    public String getName() {
        return name;
    }
}

public String getCarInsuranceName(Person person) {
    return Optional.of(person)
            .flatMap(Person::getCarAsOptional)
            .flatMap(Car::getInsuranceAsOptional)
            .map(Insurance::getName)
            .orElse("other value");
}

 

Optional 클래스는 필드 형식으로 사용할 것을 가정하지 않아, Serializable 인터페이스를 구현하지 않습니다.
때문에, 위와 같이 get 메서드에만 Optional 를 사용하는것을 권장합니다.

 

5) Optional 스트림 조작

 

자바 9 에서는 Optional 의 스트림 처리를 제공하기 위해 Optional에 stream() 메서드를 제공합니다.

 

아래는 Optional의 stream 함수를 사용한 예제입니다.

 

public Set<String> getCarInsuranceNames(List<Person> persons) {
    return persons.stream()
            .map(Person::getCarAsOptional)
            .map(optCar -> optCar.flatMap(Car::getInsuranceAsOptional))
            .map(optIns -> optIns.map(Insurance::getName))
            .flatMap(Optional::stream)
            .collect(Collectors.toSet());
}

 

stream 메서드를 지원함으로 위와 같이 더욱 간단하게 null 처리를 할 수 있게 되었습니다.

 

하지만, 마지막 결과를 얻기 위해서는 빈 Optional 은 제거하고 있는것은 언랩해야 하는 문제가 있습니다.

 

이는 위 예제에서 Optional::stream 메서드로 해결할 수 있습니다.

stream 메서드는 값이 있는 것 만을 Stream에 담아서 전달하기 때문입니다.

 

 

6) 디폴트 액션과 Optional 언랩

 

아래는 Optional 클래스가 가지고 있는 디폴트 액션입니다.

 

디폴트 액션 설명
get Optional 의 값을 가져오는 메서드입니다.
만약, 값이 없다면 NoSuchElementException이 발생하기 때문에 위험한 메서드입니다.
orElse(T other) orElse는 값이 없는경우 인자인 other를 반환합니다.
orElseGet(Supplier<? extends T> other) orElseGet는 orElse의 게으른 버전입니다.
값이 없는경우에서야 Supplier 를 수행하여 값을 반환하기 때문입니다.
orElseThrow(Supplier<? extends T> exceptionSupplier) orElseThrow는 값이 없는 경우 예외를 발생합니다.
ifPresend(Consumer<? super T> consumer) ifPresend는 값이 존재할때만, 인자의 Consumer를 수행합니다.
ifPresendOrElse(Consumer<? super T> action, Runnable emptyAction) ifPresendOrElse는 자바 9에서 추가된 메서드로,
위의 ifPresent와의 차이점으로는 값이 비어있는 경우 Runnable 인자를 실행한다는 점입니다.

 

7) 필터로 특정값 거르기

 

Optional은 filter 메서드를 지원하고 있습니다.

 

이 메서드는 프레디케이트를 인자로 받으며, Optional 객체가 값을 가지고 있는 경우 프레디 케이트를 적용하고, 값이 없는경우에는 빈 Optional를 반환합니다.

프레디 케이트 적용 결과가 false의 경우에도 빈 Optional 을 반환합니다.

 

아래는 filter 사용 예제 입니다.

 

Optional<Insurance> optInsurance = Optional.ofNullable(insurance);
optInsurance.filter(insurance -> "CabridgeInsurance".equals(insurance.getName()))
            .ifPresent(x -> System.out.println("ok"));

 

6. 마무리

 

이번 포스팅에서는 Chapter11 null 대신 Optional 클래스에 대해 진행하였습니다.

다음에는 Chapter12 새로운 날짜와 시간 API에 대해 포스팅하겠습니다.

 

반응형

+ Recent posts