Java 8의 Optional.ifPresent 및 if-not-present의 기능적 스타일?


274

Java 8에서는 Optional객체가 있으면 객체에 무언가를 하고 싶지 않으면 다른 일을 하고 싶습니다 .

if (opt.isPresent()) {
  System.out.println("found");
} else {
  System.out.println("Not found");
}

그러나 이것은 '기능적 스타일'이 아닙니다.

OptionalifPresent()방법을하지만, 나는 체인 드릴 수 없습니다 orElse()방법.

따라서 나는 쓸 수 없다 :

opt.ifPresent( x -> System.out.println("found " + x))
   .orElse( System.out.println("NOT FOUND"));

@assylias에 대한 답변으로 Optional.map()다음과 같은 경우에는 효과 가 없다고 생각 합니다.

opt.map( o -> {
  System.out.println("while opt is present...");
  o.setProperty(xxx);
  dao.update(o);
  return null;
}).orElseGet( () -> {
  System.out.println("create new obj");
  dao.save(new obj);
  return null;
});

이 경우 opt존재하는 경우 속성을 업데이트하고 데이터베이스에 저장합니다. 사용할 수 없으면 새로 작성 obj하여 데이터베이스에 저장합니다.

두 개의 람다에 주목해야한다 null.

그러나 opt존재하는 경우 두 람다가 모두 실행됩니다. obj업데이트되고 새 개체가 데이터베이스에 저장됩니다. 이것은 return null첫 번째 람다 때문입니다 . 그리고 orElseGet()계속 실행될 것입니다.


53
첫 번째 샘플을 사용하십시오. 그것은이다 아름다운 .
Sotirios Delimanolis

3
해당 동작을 위해 설계되지 않은 API를 사용할 때 특정 동작을 강제로 중단하는 것이 좋습니다. 첫 번째 예는 작은 스타일의 말과는 별개로 나에게 잘 어울리지 만, 그 말은 무시됩니다.
skiwi

4
@smallufo : 교체 return null;와 함께 return o;(모두). 그러나 나는 당신이 잘못된 곳에서 일하고 있다는 강한 느낌을 가지고 있습니다. 당신은 그것을 생산 한 현장에서 일해야합니다 Optional. 그 장소에는 중간 단계없이 원하는 작업을 수행 할 수있는 방법이 있어야합니다 Optional.
Holger

10
Java 9는 문제에 대한 솔루션을 구현합니다. iteratrlearning.com/java9/2016/09/05/java9-optional.html
pisaruk

2
이것이 쉽지 않은 이유는 의도적이라고 생각합니다. 선택 사항은 흐름 제어가 아니라 값 변환을 수행해야합니다. 나는 ifPresent이것 과 모순되는 것을 안다 . 다른 모든 방법은 조치가 아닌 가치를 참조합니다.
AlikElzin-kilaka

답변:


109

저에게 @Dane White의 대답은 괜찮습니다. 먼저 Runnable을 사용하는 것을 좋아하지 않았지만 대안을 찾을 수 없었습니다. 여기에서 선호하는 다른 구현

public class OptionalConsumer<T> {
    private Optional<T> optional;

    private OptionalConsumer(Optional<T> optional) {
        this.optional = optional;
    }

    public static <T> OptionalConsumer<T> of(Optional<T> optional) {
        return new OptionalConsumer<>(optional);
    }

    public OptionalConsumer<T> ifPresent(Consumer<T> c) {
        optional.ifPresent(c);
        return this;
    }

    public OptionalConsumer<T> ifNotPresent(Runnable r) {
        if (!optional.isPresent()) {
            r.run();
        }
        return this;
    }
}

그런 다음 :

Optional<Any> o = Optional.of(...);
OptionalConsumer.of(o).ifPresent(s ->System.out.println("isPresent "+s))
            .ifNotPresent(() -> System.out.println("! isPresent"));

업데이트 1 :

가치가 있고 그것을 처리하고 싶을 때 전통적인 개발 방식에 대한 위의 솔루션이지만 기능을 정의하고 실행하려면 다음 개선 사항을 확인하십시오.

public class OptionalConsumer<T> implements Consumer<Optional<T>> {
private final Consumer<T> c;
private final Runnable r;

public OptionalConsumer(Consumer<T> c, Runnable r) {
    super();
    this.c = c;
    this.r = r;
}

public static <T> OptionalConsumer<T> of(Consumer<T> c, Runnable r) {
    return new OptionalConsumer(c, r);
}

@Override
public void accept(Optional<T> t) {
    if (t.isPresent()) {
        c.accept(t.get());
    }
    else {
        r.run();
    }
}

그런 다음 다음과 같이 사용할 수 있습니다.

    Consumer<Optional<Integer>> c=OptionalConsumer.of(System.out::println, ()->{System.out.println("Not fit");});
    IntStream.range(0, 100).boxed().map(i->Optional.of(i).filter(j->j%2==0)).forEach(c);

이 새로운 코드에는 3 가지가 있습니다.

  1. 기존 객체를 쉽게 사용하기 전에 기능을 정의 할 수 있습니다.
  2. 각 옵션에 대해 객체 굴절을 생성하지 않음 선택 사항, 하나만, 메모리가 적고 GC가 적습니다.
  3. 다른 구성 요소와의 더 나은 사용을 위해 소비자를 구현하고 있습니다.

그건 그렇고 지금은 그 이름이 더 묘사 적입니다. 실제로 소비자입니다.>


3
Optional.of (o) 대신 Optional.ofNullable (o)을 사용해야합니다
traeper

2
사용할 값이 null인지 아닌지 확실하지 않고 NPE를 마주 칠 필요가없는 경우, null이 아니거나 확실하지 않은 경우에는 Nullable을 사용해야합니다. NPE.
Bassem Reda Zohdy

1
OptionalConsumer 클래스는 코드의 if / else보다 더 좋아 보인다고 생각합니다. 감사! :)
witek1902

207

Java 9 이상을 사용하는 경우 다음 ifPresentOrElse()방법 을 사용할 수 있습니다 .

opt.ifPresentOrElse(
   value -> System.out.println("Found: " + value),
   () -> System.out.println("Not found")
);

3
스칼라의 패턴 매칭만큼 깨끗하기 때문에
좋습니다

그런 두 람다는 꽤 못 생겼어요. 나는이 경우에 / 다른 것이 훨씬 깨끗하다고 ​​생각합니다.
john16384

1
@ john16384 좋아요, 못 생겼다면 답을 삭제하겠습니다 (아니요).
ZhekaKozlov

이것은 매우 좋지만 ifPresentOrElse를 사용할 수 없기 때문에 JDK8에 대한 질문이었습니다.
hreinn

81

자바 9 소개

ifPresentOrElse 값이 있으면 값으로 지정된 조치를 수행하고, 그렇지 않으면 주어진 빈 기반 조치를 수행합니다.

Java 8 치트 시트의 탁월한 Optional을 참조하십시오 .

대부분의 사용 사례에 대한 모든 답변을 제공합니다.

아래 요약

ifPresent ()-Optional이 설정되면 무언가를한다

opt.ifPresent(x -> print(x)); 
opt.ifPresent(this::print);

filter ()-특정 선택적 값을 거부 (필터 아웃)합니다.

opt.filter(x -> x.contains("ab")).ifPresent(this::print);

map ()-존재하는 경우 값을 변환

opt.map(String::trim).filter(t -> t.length() > 1).ifPresent(this::print);

orElse () / orElseGet () – 비어 있음

int len = opt.map(String::length).orElse(-1);
int len = opt.
    map(String::length).
    orElseGet(() -> slowDefault());     //orElseGet(this::slowDefault)

orElseThrow ()-비어있는 경우 예외적으로 예외 처리

opt.
filter(s -> !s.isEmpty()).
map(s -> s.charAt(0)).
orElseThrow(IllegalArgumentException::new);

66
이것은 실제로 OP의 질문에 대답하지 않습니다. OP는 일반적인 용도로 많이 사용하지만 OP가 요구 한 것은 아닙니다.
캡틴 맨

1
@CaptainMan은 실제로 그렇게합니다. opt.map ( "found"). orElse ( "not found") 표현식이 청구서를 채 웁니다.
Matt

4
@ 매트 아니요, OP는 선택 사항이 있거나 없을 때 수행 할 작업을 요청하고 값이 있거나 없을 때 값을 반환하지 않도록 요청합니다. OP는 orElseGet을 사용하여 왜 작동하지 않는지 설명하는 질문에서 비슷한 것을 언급합니다.
Captain Man

2
@CaptainMan 당신의 요점을 참조하십시오. 그가에서 null을 반환하지 않으면 작동 할 수 있다고 생각 map하지만 DAO를 호출 할 수있는 기능 솔루션을 요청하는 것은 약간 이상합니다. 이 map.orElse블록 에서 업데이트 된 / 새 객체를 반환 한 다음 반환 된 객체로 필요한 작업을 수행하는 것이 더 합리적 입니다.
Matt

1
내 생각 map스트림 자체에 초점이 "스트림이 요소의 상태에 따라 다른 물체에 일을"하기위한 것이 아닙니다. 그것을 알고 좋은 ifPresentOrElse자바 9에 추가
WesternGun

53

대안은 다음과 같습니다.

System.out.println(opt.map(o -> "Found")
                      .orElse("Not found"));

그래도 가독성이 향상되지는 않는다고 생각합니다.

또는 Marko가 제안한 것처럼 삼항 연산자를 사용하십시오.

System.out.println(opt.isPresent() ? "Found" : "Not found");

2
감사합니다 @assylias이지만 Optional.map ()이 사례에서 작동하지 않는다고 생각합니다 (컨텍스트 업데이트 참조).
smallufo

2
@smallufo 당신은 new Object();첫 번째 람다 로 돌아 가야 하지만 정직하게 말하면 매우 추악합니다. 업데이트 된 예제의 경우 if / else를 고수합니다.
assylias

동의 하면 체인에 map반환하기 만하면 Optional코드를 이해하기 어렵지만 map문자 그대로 무언가를 매핑한다고 가정합니다.
Tiina

40

또 다른 해결책은 다음과 같이 고차 함수를 사용하는 것입니다.

opt.<Runnable>map(value -> () -> System.out.println("Found " + value))
   .orElse(() -> System.out.println("Not Found"))
   .run();

8
내 눈에 JDK 9가없는 지금까지 최고의 솔루션
Semaphor

5
설명이 좋을 것입니다. 나는 왜 당신이 런닝 가능한 맵 (?)을 사용 해야하는지 그리고 그 value -> () -> syso부분의 의미를 묻습니다 .
froehli

이 솔루션에 감사드립니다! Runnable을 사용하는 이유는 out map이 어떤 값도 반환하지 않으며 Runnable을 사용하면 람다를 반환하고 map의 결과는 람다와 같이 실행한다는 것입니다. 따라서 값을 반환하면 다음을 사용할 수 있습니다.String result = opt.map(value -> "withOptional").orElse("without optional");
nanotexnik

21

즉시 사용할 수있는 좋은 방법은 없습니다. 클리너 구문을 정기적으로 사용하려면 다음과 같은 유틸리티 클래스를 작성할 수 있습니다.

public class OptionalEx {
    private boolean isPresent;

    private OptionalEx(boolean isPresent) {
        this.isPresent = isPresent;
    }

    public void orElse(Runnable runner) {
        if (!isPresent) {
            runner.run();
        }
    }

    public static <T> OptionalEx ifPresent(Optional<T> opt, Consumer<? super T> consumer) {
        if (opt.isPresent()) {
            consumer.accept(opt.get());
            return new OptionalEx(true);
        }
        return new OptionalEx(false);
    }
}

그런 다음 다른 곳에서 정적 가져 오기를 사용하여 다음에 가까운 구문을 얻을 수 있습니다.

import static com.example.OptionalEx.ifPresent;

ifPresent(opt, x -> System.out.println("found " + x))
    .orElse(() -> System.out.println("NOT FOUND"));

감사. 이 솔루션은 아름답습니다. 내장 솔루션이 없을 수도 있습니다 (JDK가 그러한 방법을 통합하지 않는 한). OptionalEx는 매우 유용합니다. 아무튼 감사 해요.
smallufo

예, 결과가 마음에 들며 스타일이 지원됩니다. 그렇다면 표준 API가 아닌 이유는 무엇입니까?
guthrie

좋은 대답입니다. 우리도 마찬가지입니다. API (또는 언어)에 있어야하지만 동의하지 않았습니다 : bugs.openjdk.java.net/browse/JDK-8057557 .
개렛 스미스

좋은. 이는 고려할 JDK 8.1의 일부 여야합니다.
peter_pilgrim

35
Optional.ifPresentOrElse()JDK 9에 추가되었습니다.
Stuart Marks

9

Java 8 이하 만 사용할 수있는 경우 :

1) spring-data지금까지 가장 좋은 방법 이 없다면 :

opt.<Runnable>map(param -> () -> System.out.println(param))
      .orElse(() -> System.out.println("no-param-specified"))
      .run();

이제는 읽기가 어렵고 이해하기가 어렵다는 것을 알고 있지만 개인적으로 나에게 잘 어울리 며이 경우에는 유창한 또 다른 유창한 방법을 보지 못합니다.

2) 당신이 충분히 운 spring-data이 좋으며 가장 좋은 방법을 사용할 수 있다면 Optionals #ifPresentOrElse입니다 .

Optionals.ifPresentOrElse(opt, System.out::println,
      () -> System.out.println("no-param-specified"));

Java 9를 사용할 수 있다면 반드시 다음을 수행해야합니다.

opt.ifPresentOrElse(System.out::println,
      () -> System.out.println("no-param-specified"));

2

설명 된 동작은 대부분의 Scala 구문을 구현하는 Java 8+ 용 객체 기능 라이브러리 인 Vavr (이전의 Javaslang) 을 사용하여 달성 할 수 있습니다 (Scala는 더 풍부한 유형 시스템을 JVM에 구축 한보다 표현적인 언어 임). 순수 기능 코드를 작성하기 위해 Java 프로젝트에 추가하는 것은 매우 좋은 라이브러리입니다.

Vavr은 다음 Option과 같은 옵션 유형으로 작동하는 기능을 제공하는 모나드를 제공합니다.

  • fold: 두 경우 모두 옵션 값을 맵핑합니다 (정의 / 빈).
  • onEmpty: Runnable옵션이 비어있을 때 실행할 수 있습니다
  • peek: 옵션 값을 정의 할 수 있습니다 (정의 된 경우).
  • 또한 Serializable반대로 Optional메소드 인수 및 인스턴스 멤버로 안전하게 사용할 수 있습니다.

Option은 Java의 Optional "pseudo-monad"와 다른 모나드 법칙을 따르며보다 풍부한 API를 제공합니다. 물론 Java의 Optional (및 다른 방법으로)에서이를 작성할 수 있습니다. Option.ofOptional(javaOptional)–Vavr은 상호 운용성에 중점을 둡니다.

예제로 이동 :

// AWESOME Vavr functional collections (immutable for the gread good :)
// fully convertible to Java's counterparts.
final Map<String, String> map = Map("key1", "value1", "key2", "value2");

final Option<String> opt = map.get("nonExistentKey"); // you're safe of null refs!

final String result = opt.fold(
        () -> "Not found!!!",                // Option is None
        val -> "Found the value: " + val     // Option is Some(val)
);

추가 자료

널 참조, 억 달러의 실수

NB 이것은 Vavr이 제공하는 것의 아주 작은 예일뿐입니다.


1

다른 해결책은 다음과 같습니다.

이것은 당신이 그것을 사용하는 방법입니다 :

    final Opt<String> opt = Opt.of("I'm a cool text");
    opt.ifPresent()
        .apply(s -> System.out.printf("Text is: %s\n", s))
        .elseApply(() -> System.out.println("no text available"));

또는 반대 사용 사례의 경우에 해당하는 경우 :

    final Opt<String> opt = Opt.of("This is the text");
    opt.ifNotPresent()
        .apply(() -> System.out.println("Not present"))
        .elseApply(t -> /*do something here*/);

이것은 재료입니다 :

  1. "elseApply"메소드만을위한 거의 수정 된 기능 인터페이스
  2. 선택적 향상
  3. 약간의 curring :-)

"화장품"강화 기능 인터페이스.

@FunctionalInterface
public interface Fkt<T, R> extends Function<T, R> {

    default R elseApply(final T t) {
        return this.apply(t);
    }

}

그리고 향상을위한 선택적 래퍼 클래스 :

public class Opt<T> {

    private final Optional<T> optional;

    private Opt(final Optional<T> theOptional) {
        this.optional = theOptional;
    }

    public static <T> Opt<T> of(final T value) {
        return new Opt<>(Optional.of(value));
    }

    public static <T> Opt<T> of(final Optional<T> optional) {
        return new Opt<>(optional);
    }

    public static <T> Opt<T> ofNullable(final T value) {
        return new Opt<>(Optional.ofNullable(value));
    }

    public static <T> Opt<T> empty() {
        return new Opt<>(Optional.empty());
    }

    private final BiFunction<Consumer<T>, Runnable, Void> ifPresent = (present, notPresent) -> {
        if (this.optional.isPresent()) {
            present.accept(this.optional.get());
        } else {
            notPresent.run();
        }
        return null;
    };

   private final BiFunction<Runnable, Consumer<T>, Void> ifNotPresent = (notPresent, present) -> {
        if (!this.optional.isPresent()) {
            notPresent.run();
        } else {
            present.accept(this.optional.get());
        }
        return null;
    };

    public Fkt<Consumer<T>, Fkt<Runnable, Void>> ifPresent() {
        return Opt.curry(this.ifPresent);
    }

    public Fkt<Runnable, Fkt<Consumer<T>, Void>> ifNotPresent() {
        return Opt.curry(this.ifNotPresent);
    }

    private static <X, Y, Z> Fkt<X, Fkt<Y, Z>> curry(final BiFunction<X, Y, Z> function) {
        return (final X x) -> (final Y y) -> function.apply(x, y);
    }
}

이것은 트릭을 수행해야하며 이러한 요구 사항을 처리하는 방법을 기본 템플릿으로 사용할 수 있습니다.

기본 아이디어는 다음과 같습니다. 비 기능적 스타일의 프로그래밍 세계에서는 아마도 두 매개 변수를 사용하는 메서드를 구현할 것입니다. 첫 번째 매개 변수는 값을 사용할 수있는 경우 실행 해야하는 실행 가능한 코드의 종류이고 다른 매개 변수는 실행 가능한 코드입니다. 값을 사용할 수 없습니다. 가독성을 높이기 위해 curring을 사용하여 두 매개 변수의 기능을 한 매개 변수의 두 기능으로 각각 분할 할 수 있습니다. 이것이 기본적으로 내가 한 일입니다.

힌트 : Opt은 값을 사용할 수없는 경우를 대비하여 코드를 실행하려는 다른 사용 사례도 제공합니다. 이것은 Optional.filter.stuff를 통해서도 가능하지만 훨씬 더 읽기 쉽습니다.

희망이 도움이됩니다!

좋은 프로그래밍 :-)


작동하지 않는 것을 알 수 있습니까? 다시 테스트했는데 제대로 작동합니까?
Alessandro Giusa

실례지만 ckass Fkt는 어디에 정의되어 있습니까? 마지막으로, 컴파일 문제가 있습니다 : 오류 : (35, 17) java : 변수 옵션이 초기화되지 않았을 수 있습니다
Enrico Giurin

Fkt는 위에서 인터페이스로 정의됩니다. 전체 기사를 읽으십시오 :-)
Alessandro Giusa

예. EFunction 인터페이스를 이름으로 Fkt로 변경했습니다. 오타가있었습니다. 리뷰 주셔서 감사합니다 :-) 죄송합니다.
Alessandro Giusa

나는 그런 코드를 작성하는 것이 좋은 생각이라고 생각하지 않습니다 ... 가능하면 jdk 유틸리티를 사용해야합니다.
Tyulpan Tyulpan

0

값을 저장하려는 경우 :

Pair.of<List<>, List<>> output = opt.map(details -> Pair.of(details.a, details.b))).orElseGet(() -> Pair.of(Collections.emptyList(), Collections.emptyList()));

0

목록이 있고 isPresent()문제 (선택 사항과 관련됨)를 피한다고 가정하면 .iterator().hasNext()존재하지 않는지 확인할 수 있습니다 .

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.