Optional

Java 8์— ์ƒˆ๋กœ ์ƒ๊ธด ์ธํ„ฐํŽ˜์ด์Šค๋กœ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ ๋ฉ”์„œ๋“œ๊ฐ€ ๋ฐ˜ํ™˜ํ•  ๊ฒฐ๊ณผ๊ฐ’์ด ์—†์Œ์„ ๋ช…๋ฐฑํ•˜๊ฒŒ ํ‘œํ˜„ํ•  ํ•„์š”๊ฐ€ ์žˆ๋Š” ๊ณณ์—์„œ ์ œํ•œ์ ์œผ๋กœ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋Š” ๋ฉ”์ปค๋‹ˆ์ฆ˜์„ ์ œ๊ณตํ•˜๊ธฐ ์œ„ํ•ด ์ƒˆ๋กœ ์ƒ๊ฒจ๋‚ฌ๋‹ค.

Java api doc์˜ API ๋…ธํŠธ๋ฅผ ๋ณด๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์ด ์„ค๋ช…ํ•˜๊ณ  ์žˆ๋‹ค. Optional์€ ์ฃผ๋กœ ๊ฒฐ๊ณผ ์—†์Œ์„ ๋‚˜ํƒ€๋‚ผ ํ•„์š”์„ฑ์ด ๋ช…ํ™•ํ•˜๊ณ  null์„ ์‚ฌ์šฉํ•˜๋ฉด ์˜ค๋ฅ˜๊ฐ€ ๋ฐœ์ƒํ•  ์ˆ˜ ์žˆ๋Š” ๋ฉ”์†Œ๋“œ ๋ฐ˜ํ™˜ ์œ ํ˜•์œผ๋กœ ์‚ฌ์šฉํ•˜๋„๋ก ๊ณ ์•ˆ๋˜์—ˆ๋‹ค. ์œ ํ˜•์ด ์˜ต์…˜์ธ ๋ณ€์ˆ˜ ์ž์ฒด๋Š” null์ด ์•„๋‹ˆ๋ฏ€๋กœ ํ•ญ์ƒ ์˜ต์…˜ ์ธ์Šคํ„ด์Šค๋ฅผ ๊ฐ€๋ฆฌ์ผœ์•ผ ํ•œ๋‹ค.

Optional์„ ์˜ฌ๋ฐ”๋ฅด๊ฒŒ ์‚ฌ์šฉํ•˜๊ธฐ

1. Optional ๋ณ€์ˆ˜์— null ๋Œ€์‹  Optional.empty() ์‚ฌ์šฉ

Optional<Car> optionalCar = null;

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

2. Optional ๊ฐ’์„ ๊บผ๋‚ด์“ฐ๊ธฐ ์ „์— ๊ฐ’์ด ์žˆ๋Š”์ง€ ํ™•์ธํ•˜๊ธฐ

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

Car car = optionalCar.get(); //NoSuchElementException ๋ฐœ์ƒ

Optional์€ ๊ฐ’์ด ๋น„์–ด์žˆ์„ ์ˆ˜๋„ ์žˆ๊ธฐ ๋•Œ๋ฌธ์— ์‚ฌ์šฉํ•˜๊ธฐ์ „์— ์กด์žฌํ•œ๋‹ค๋Š” ๊ฒƒ์„ ์ฆ๋ช…ํ•ด์•ผ ํ•˜๋Š”๋ฐ ์ผ๋ฐ˜์ ์œผ๋กœ๋Š” isPresent()ํ›„์— get()์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ์ง€๋งŒ ์ฝ”๋“œ๋„ ๊ธธ์–ด์ง€๊ณ  ํ•œ๋ฒˆ์— ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋Š” API๋ฅผ ์ œ๊ณตํ•˜๊ธฐ ๋•Œ๋ฌธ์— ์ด๋ฅผ ์‚ฌ์šฉํ•ด๋ณด์ž.

//isPresent - get
if(optionalCar.isPresent()){
    return optionalCar.get();
}else {
    return null;
}

//orElse
return optionalCar.orElse(null);

//orElseGet
return optionalCar.orElseGet(() -> null);

orElseGet()์€ Supplier๋ฅผ ์ธ์ž๋กœ ๋ฐ›์œผ๋ฉฐ, ๊ฐ’์ด ์—†์„๋•Œ์— ํ•ด๋‹น supplier๊ฐ€ ์ˆ˜ํ–‰๋œ๋‹ค. ํ•˜์ง€๋งŒ orElse()๋Š” Optional๋กœ ๊ฐ์‹ธ๊ณ  ์žˆ๋Š” ๊ฐ์ฒดํƒ€์ž…์„ ์ธ์ž๋กœ ๋ฐ›์œผ๋ฉฐ ๊ฐ’์ด ์žˆ๋”๋ผ๋„ ๋‚ด๋ถ€๊ฐ€ ์ˆ˜ํ–‰๋˜๊ณ  ์‚ฌ์šฉ๋˜์ง€ ์•Š๋Š” ๊ฒฝ์šฐ ํ•ด๋‹น ๊ฐ์ฒด๋ฅผ ์ง€์šฐ๊ฒŒ ๋˜์–ด ํ•„์š”์—†๋Š” ์˜ค๋ฒ„ํ—ค๋“œ๊ฐ€ ๋ฐœ์ƒํ•œ๋‹ค.

Optional<String> optStr = Optional.of("Hi");
optStr.orElse(new String("hi1"));
optStr.orElseGet(() -> new String("hi1"));

//bytecode
 L1
    LINENUMBER 9 L1
    ALOAD 1
    NEW java/lang/String
    DUP
    LDC "hi"
    INVOKESPECIAL java/lang/String.<init> (Ljava/lang/String;)V
    INVOKEVIRTUAL java/util/Optional.orElse (Ljava/lang/Object;)Ljava/lang/Object;
    POP
L2
    LINENUMBER 11 L2
    ALOAD 1
    INVOKEDYNAMIC get()Ljava/util/function/Supplier; [
      // handle kind 0x6 : INVOKESTATIC
      java/lang/invoke/LambdaMetafactory.metafactory(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
      // arguments:
      ()Ljava/lang/Object;, 
      // handle kind 0x6 : INVOKESTATIC
      study/OptionalEx.lambda$main$0()Ljava/lang/String;, 
      ()Ljava/lang/String;
    ]
    INVOKEVIRTUAL java/util/Optional.orElseGet (Ljava/util/function/Supplier;)Ljava/lang/Object;
    POP

    ...
private static synthetic lambda$main$0()Ljava/lang/String;
 L0
    LINENUMBER 11 L0
    NEW java/lang/String
    DUP
    LDC "hi"
    INVOKESPECIAL java/lang/String.<init> (Ljava/lang/String;)V
    ARETURN
    MAXSTACK = 3
    MAXLOCALS = 0
}

์˜ˆ๋ฅผ ๋“ค์–ด ์œ„์™€ ๊ฐ™์€ ์ฝ”๋“œ๋ฅผ ์ž‘์„ฑํ–ˆ์„๋•Œ optStr์€ null์ด ์•„๋‹ˆ๋ผ new String("hi1")๊ฐ€ ์‹คํ–‰๋˜์ง€ ์•Š์„ ๊ฒƒ ๊ฐ™์ง€๋งŒ ๋ฐ”์ดํŠธ์ฝ”๋“œ๋ฅผ ๋ณด๋ฉด ์ƒˆ๋กœ ๋ฌธ์ž์—ด์„ ์ƒ์„ฑํ–ˆ๋‹ค๊ฐ€ POPํ•˜๋Š” ๊ฒƒ์„ ๋ณผ ์ˆ˜ ์žˆ๊ณ , orElseGet()์€ ์šฐ๋ฆฌ๊ฐ€ lambda์—์„œ ๋ดค๋˜๊ฒƒ์ฒ˜๋Ÿผ static ๋ฉ”์„œ๋“œ๋กœ ์ƒ์„ฑํ•˜์—ฌ ํ˜ธ์ถœ๋˜๋Š” ํƒ€์ด๋ฐ์— ์ด๋ฅผ ์‹คํ–‰ํ•ด ๊ฐ์ฒด๋ฅผ ์ƒ์„ฑํ•˜๋Š” ๊ฒƒ์„ ๋ณผ์ˆ˜์žˆ์–ด orElse๋Š” ํ•„์š”์—†๋Š” ์˜ค๋ฒ„ํ—ค๋“œ์— ์ฃผ์˜ํ•ด์•ผํ•œ๋‹ค. ํ•˜์ง€๋งŒ ๋ฐ˜๋“œ์‹œ ์ด๋ฏธ ์ƒ์„ฑ๋˜์–ด์žˆ๋Š” ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋Š” ๊ฒƒ์ด๋ผ๋ฉด orElse()๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด ์ข‹์„ ์ˆ˜๋„ ์žˆ๋‹ค.

//before
if(optionalCar.isPresent()){
    return optionalCar.get();
}else {
    throw new NoSuchElementException();
}

//after
optionalCar.orElseThrow();  

์˜ˆ์™ธ์ฒ˜๋ฆฌ๋ฅผ ํ•˜๋Š” ๊ฒฝ์šฐ์—๋„ isPresent()๋ฅผ ํ†ตํ•œ ์˜ˆ์™ธ์ฒ˜๋ฆฌ๋ณด๋‹ค๋Š” orElseThrow()๋ฅผ ์ด์šฉํ•˜์—ฌ ์˜ˆ์™ธ์ฒ˜๋ฆฌ๋ฅผ ํ•˜๋Š” ๊ฒƒ์ด ๋ฐ”๋žŒ์งํ•˜๋‹ค. ์ธ์ž๋กœ Supplier๋ฅผ ํ†ตํ•ด ํŠน์ • Exception์„ ๋˜์งˆ ์ˆ˜ ์žˆ๋Š”๋ฐ ์•„๋ฌด๊ฒƒ๋„ ์ฃผ์ง€ ์•Š์œผ๋ฉด ๊ธฐ๋ณธ์ ์œผ๋กœ NoSuchElementException์„ ๋˜์ง„๋‹ค.

3. Optional์ด ์žˆ์„๋•Œ๋งŒ ์ด๋ฅผ ์†Œ๋น„ํ•˜์—ฌ ๋ฌด์–ธ๊ฐ€๋ฅผ ํ• ๋•Œ๋Š” ifPresent()๋ฅผ ํ™œ์šฉ

//before
if(optionalCar.isPresent()){
    System.out.println(optionalCar.get());
}

//after
optionalCar.ifPresent(System.out::println);

4. ์ปฌ๋ ‰์…˜์€ Optional๋Œ€์‹  ๋น„์–ด์žˆ๋Š” ์ปฌ๋ ‰์…˜์„ ์‚ฌ์šฉํ•˜์ž.

//before
List<Car> cars = carFactory.getCars();
return Optional.ofNullable(cars);

//after
List<Car> cars = carFactory.getCars();
return cars != null ? cars : Collections.emptyList();

5. ์ปฌ๋ ‰์…˜์€ Optional๋กœ ๊ฐ์‹ธ์ง€๋ง์ž.

์ปฌ๋ ‰์…˜ ์ž์ฒด๊ฐ€ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ์‹ผ ํ˜•ํƒœ์˜ ๊ฐ์ฒด์ด๊ณ  ์ด๋„ ์ถฉ๋ถ„ํ•œ API๋ฅผ ์ œ๊ณตํ•˜๊ธฐ ๋•Œ๋ฌธ์— ํ•œ๋ฒˆ๋” Optional๋กœ ๊ฐ์‹ธ๋ฉด ํ•„์š”์—†๋Š” ์˜ค๋ฒ„ํ—ค๋“œ๊ฐ€ ๋ฐœ์ƒํ•œ๋‹ค. JPA์˜ ๋ฉ”์„œ๋“œ๋ฅผ ์ƒ์„ฑํ• ๋•Œ๋„ JPA์ž์ฒด์ ์œผ๋กœ ๋น„์–ด์žˆ๋Š” ์ปฌ๋ ‰์…˜์„ ๋ฐ˜ํ™˜ํ•ด์ฃผ๋ฏ€๋กœ Optional๋กœ ๊ฐ์Œ€ ํ•„์š”๊ฐ€ ์—†๋‹ค.

//bad
Optional<List<Car>> findAllByCompanyName(String name);

//good
List<Car> findAllByCompanyName(String name);

6. ์ปฌ๋ ‰์…˜,Map์˜ ์›์†Œ๋กœ Optional์„ ์‚ฌ์šฉํ•˜์ง€ ๋ง์ž.

7. ๋‹จ์ผ ๊ฐ’์„ ์–ป๊ธฐ ์œ„ํ•œ ๋ชฉ์ ์œผ๋กœ ๋ฉ”์„œ๋“œ ์ฒด์ด๋‹์„ ํ•˜์ง€๋ง์ž.

//bad
String status = "on";
return Optional.ofNullable(status).orElse("PENDING");

//good
status == null ? "PENDING" : status;

Optional์„ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์€ ๊ฒฐ๊ตญ ๋˜ํ•˜๋‚˜์˜ ๋ž˜ํผ๊ฐ์ฒด๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด๋ฏ€๋กœ ๋‹จ์ˆœํ•œ ๋กœ์ง์ด๋ผ๋ฉด ๊ทธ๋ƒฅ ์ฝ”๋”ฉํ•˜์ž. ๊ตฌ๊ด€์ด ๋ช…๊ด€์ด๋ผํ–ˆ๋‹ค.

8. Optional์„ ํ•„๋“œ๋กœ ์‚ฌ์šฉํ•˜์ง€๋ง์ž.

public class Car {
    private Optional<Navigation> navigation = Optional.empty(); //bad 
}

Optional์€ ์• ์ดˆ์— ํ•„๋“œ๋กœ ์‚ฌ์šฉํ•  ๋ชฉ์ ์œผ๋กœ ๋งŒ๋“ค์–ด์ง€์ง€ ์•Š์•„ Serializable๋„ ๊ตฌํ˜„ํ•˜์ง€ ์•Š์•˜๊ธฐ ๋•Œ๋ฌธ์— ์‚ฌ์šฉ์„ ์ง€์–‘ํ•ด์•ผ ํ•œ๋‹ค.

9. Optional์„ ๋ฉ”์„œ๋“œ,์ƒ์„ฑ์ž ์ธ์ž๋กœ ์‚ฌ์šฉํ•˜์ง€๋ง์ž.

//bad
public void render(Optional<Renderer> renderer){
    renderer.orElseThrow(() -> new IllegalArgumentException("null ์ผ ์ˆ˜ ์—†์Šต๋‹ˆ๋‹ค."));
    //...
}

//good
public void render(Renderer renderer){
    if(renderer == null){
        throw new IllegalArgumentException("null ์ผ ์ˆ˜ ์—†์Šต๋‹ˆ๋‹ค.");
    }
    //...
}

์ด๋Ÿฌํ•œ ๋ฐฉ๋ฒ•์€ ๋ถˆํ•„์š”ํ•˜๊ฒŒ ์ฝ”๋“œ๋ฅผ ๋ณต์žกํ•˜๊ฒŒ ํ•  ๋ฟ์•„๋‹ˆ๋ผ ์ด๋ฅผ ํ˜ธ์ถœํ•˜๋Š” ์ชฝ์—์„œ๋„ Optional ์ƒ์„ฑ์„ ๊ฐ•์ œํ•˜๊ฒŒ ํ•˜๋Š” ๊ฒƒ์ด๋‹ค. ๋˜ํ•œ, Optional์€ ํ•˜๋‚˜์˜ ๊ฐ์ฒด๋กœ ์ด๋ฅผ ํ˜ธ์ถœํ•˜๋Š” ๊ฒƒ์ด ๊ฒฐ์ฝ” ๋น„์šฉ์ด ์ €๋ ดํ•˜์ง€ ์•Š๋‹ค.

10. null์ด ํ™•์‹คํ•˜๋ฉด ofNullable()์ด ์•„๋‹Œ of()๋ฅผ ์‚ฌ์šฉํ•˜์ž.

//bad
Optional.ofNullable("NULL์ผ ์ˆ˜ ์—†์ง€!");

//good
Optional.of("NULL์ผ ์ˆ˜ ์—†์ง€!");
//ofNullable
public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
}

ofNullable์€ ๋‚ด๋ถ€์ ์œผ๋กœ ๋ณด๋ฉด ์‚ผํ•ญ์—ฐ์‚ฐ์ž๋ฅผ ํ†ตํ•ด ๋น„์–ด์žˆ์ง€ ์•Š๋Š” ๊ฒฝ์šฐ of๋ฅผ ํ˜ธ์ถœํ•˜๋Š” ๊ฒƒ์„ ๋ณผ ์ˆ˜์žˆ๋‹ค. ๊ทธ๋ ‡๊ธฐ ๋•Œ๋ฌธ์— ์ด๋Ÿฌํ•œ ์—ฐ์‚ฐ์„ ์กฐ๊ธˆ์ด๋ผ๋„ ์ค„์ผ ์ˆ˜ ์žˆ๊ธฐ ๋•Œ๋ฌธ์— of๋ฅผ ์‚ฌ์šฉํ•˜์ž.

11. Optional์˜ ํƒ€์ž…์ด Primitiveํƒ€์ž…์ด๋ฉด OptionalInt,OptionalLong, OptionalDouble์„ ์‚ฌ์šฉํ•˜์ž

//bad
Optional<Integer> max = Optional.of(10);
for(int i=0; i < max.get(); i++)

//good
OptionalInt max = OptionalInt.of(10);
for(int i=0; i < max.getAsInt(); i++)

//byte code
L0
    LINENUMBER 10 L0
    BIPUSH 10
    INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;
    INVOKESTATIC java/util/Optional.of (Ljava/lang/Object;)Ljava/util/Optional;
    ASTORE 0
L1
    LINENUMBER 12 L1
    BIPUSH 10
    INVOKESTATIC java/util/OptionalInt.of (I)Ljava/util/OptionalInt;
    ASTORE 1

๋‚ด๋ถ€์ ์œผ๋กœ Integer.valueOf()๋ฅผ ํ†ตํ•ด ํ•œ๋ฒˆ boxing์ด ์ผ์–ด๋‚˜๋Š” ๊ฒƒ์„ ๋ณผ ์ˆ˜ ์žˆ๊ณ  ๋˜ ์ด๋ฅผ ์‚ฌ์šฉํ• ๋•Œ unboxing์ด ์ผ์–ด๋‚˜๊ธฐ ๋•Œ๋ฌธ์— OptionalInt๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด ์ข‹๋‹ค.

12. Optional์„ ๋ฆฌํ„ดํ•˜๋Š” ๋ฉ”์„œ๋“œ์—์„œ null์„ ๋ฆฌํ„ดํ•˜์ง€ ๋ง์ž.

public static Optional<String> hi(){
        return null;
}

๋‹น์—ฐํ•œ๊ฑฐ์ง€๋งŒ Optional๋„ ๊ฐ์ฒด์ด๊ธฐ ๋•Œ๋ฌธ์— null์„ ๋ฆฌํ„ด์ด ๊ฐ€๋Šฅํ•œ๋ฐ ์ด๋ ‡๊ฒŒ ๋ฆฌํ„ดํ•˜๊ฒŒ ๋˜๋ฉด Optional์„ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด ์˜๋ฏธ๊ฐ€ ์—†๊ธฐ ๋•Œ๋ฌธ์— null์„ ๋ฆฌํ„ดํ•˜์ง€ ๋ง์ž.


Reference

https://dzone.com/articles/using-optional-correctly-is-not-optional

Last updated