Java
[Java] Java8, Optional 이란?
kyunge_ev
2022. 11. 11. 22:09
🌈 Optional
개발 시 가장 많이 발생하는 예외인 NPE(NullPointerException)을 피하기 위해 Null 여부를 검사하는데 Null 검사를 해야하는 변수가 많아 질 경우 코드가 복잡해지고 많아진다. 그래서 Null 대신 초기값을 사용하길 권장하는데 이 때, 사용하는 것을 Optional 이라고 한다.
- JAVA8 부터 Optional<T> 클래스를 사용해 NPE를 방지할 수 있도록 도와줌
- Optional 클래스는 제너릭(Generic)으로 값의 타입을 지정
- Optional 클래스는 Null이 올 수 있는 값을 감싸는 래퍼(Wrapper) 클래스
- Null 이더라도 바로 NPE가 발생하지 않음 👉 NPE(NullPointerException)를 방지
- 클래스이기 때문에 각종 메소드를 제공해줌
- .orElseThrow() / .orElse 등 편의기능을 사용하여 코드를 간결하게 만들 수 있다.
- Optional은 값을 Wrapping 하고 다시 풀때 null일 경우 대체하는 함수를 호출
→ 오버헤드 발생 할 수 있으므로 잘 못 사용 시 시스템 성능 저하
→ 메서드의 반환 값이 절대 null이 아니라면 Optional 사용을 권장하지 않음
🌈 Optional 생성
📚 Optional.empty() ⇒ 값이 Null인 경우
- 값이 null이라도 NPE를 발생시키지 않고, Optional.empty를 리턴
- Optional 클래스 내부에서 static 변수로 empty 객체를 미리 생성하여 가지고 있음
👉 빈 객체를 여러번 생성해줘야 하는 경우에도 1개의 empty 객체를 공유함으로써 메모리를 절약함
1
2
3
4
|
Optional<String> optional = Optional.empty();
System.out.println(optional); // 출력 : Optional.empty
System.out.println(optional.isPresent()); // 출력 : false
|
cs |
📚 Optional.of() ⇒ 값이 Null이 아닌 경우
- 값이 절대 null일 수 없는 경우 사용, 만약 null로 저장할 경우 NullPointerException 발생
1
|
Optional<String> optional = Optional.of("MyName");
|
cs |
📚 Optional.ofNullbale() ⇒ 값이 Null일 수도, 아닐 수도 있는 경우
- orElse , orElseGet 메서드를 사용하여 값이 없는 경우라도 안전하게 값을 가져올 수 있음
1
2
3
4
|
Optional<String> optional = Optional.ofNullable(getName());
// 값이 없다면 "Null"를 리턴
String name = optional.orElse("Null");
|
cs |
🌈 Optional이 제공하는 메소드
📃 .filter()
- filter 메소드의 인자인 람다식이 true이면 Optional 객체를 그대로 통과시키고, false라면 Optional.emty()를 return하여 추가로 처리되지 않도록 함.
1
2
3
4
5
|
String result1 = Optional.of("optional").filter((val) -> val.contains("opt")).orElse("Does not contain opt");
System.out.println(result1); //optional
String result2 = Optional.of("optional").filter((val) -> val.contains("abc")).orElse("Does not contain abc");
System.out.println(result2); //Does not contain abc
|
cs |
📃 .map()
- 입력받은 값을 다른 값으로 변환하는 메소드
- abcde를 대문자 ABCDE로 변환
1
2
|
String result3 = Optional.of("abcde").map(String::toUpperCase).orElse("fail");
System.out.println(result3); //ABCDE
|
cs |
📃 .isPresent()
- 값이 존재한다면 true를 반환하고 존재하지 않으면 false를 반환
1
2
|
if (optionalLike.isPresent()){ // like가 존재한다면,
likeRepository.delete(optionalLike.get()); // like를 삭제한다.
|
cs |
📃 .ifPresent()
- 람다식을 인자로 받는데, 값이 존재할 때만 람다식이 적용되고, 값이 존재하지 않으면 실행되지 않는다.
1
2
3
|
Optional.of("ABCDE").ifPresent(System.out :: println); //결과 : ABCDE
Optional.ofNullable(null).ifPresent(System.out :: println); //결과 : 아무것도 출력되지 않음
|
cs |
📃 .get()
- Optional 객체가 가지고 있는 value를 가지고옴 👉 만약 객체가 없다면, NoSuchElementException이 발생
1
2
|
if (optionalLike.isPresent()){ // like가 존재한다면,
likeRepository.delete(optionalLike.get()); // like를 삭제한다.
|
cs |
📃 .orElse()
- Optional 객체가 Null이라면 orElse() 메소드가 지정된 값으로 return
📃 .orElseThrow()
- 연산을 끝낸 후에도 Optional 객체가 Null이라면 예외 공급자 함수를 통해 예외를 발생
1
2
3
4
5
6
7
8
9
10
11
12
|
public CommentDeleteResponse delete(Long postsId, Long id, String userName) {
// 게시물이 존재하지 않는 경우
Post post = postRepository.findById(postsId)
.orElseThrow(() -> new AppException(ErrorCode.POST_NOT_FOUND, ErrorCode.POST_NOT_FOUND.getMessage()));
// ...
return CommentDeleteResponse.builder()
.message("포스트 삭제 완료")
.id(postsId)
.build();
}
|
cs |
📢
그 외, Java9에서 추가된 or(), isPresentOrElse(), stream() 메소드와
Java10에서 추가된 인자를 받지 않는 orElseThrow() 메소드가 있다.
(추가로 블로그 수정예정)