ProgramingTip

옵션이없는 경우 실행하는 방법은 무엇입니까?

bestdevel 2020. 11. 26. 19:42
반응형

옵션이없는 경우 실행하는 방법은 무엇입니까?


java8을 사용하여 다음 코드를 바꾸고 싶습니다 Optional.

public Obj getObjectFromDB() {
    Obj obj = dao.find();
    if (obj != null) {
        obj.setAvailable(true);
    } else {
        logger.fatal("Object not available");
    }

    return obj;
}

다음 의사 코드는 orElseRun방법 이 없기 때문에 하지만 비록 내 목적을 보여줍니다.

public Optional<Obj> getObjectFromDB() {
    Optional<Obj> obj = dao.find();
    return obj.ifPresent(obj.setAvailable(true)).orElseRun(logger.fatal("Object not available"));
}

Java 9 이상을 사용하면 다음과 같이 할 수 있습니다.ifPresentOrElse

Optional<> opt = dao.find();

opt.ifPresentOrElse(obj -> obj.setAvailable(true),
                    () -> logger.error("…"));

vavr 등을 사용하여 커링 하면 더 깔끔한 코드를 얻을 수 있습니다 아직 시도하지 않습니다.


나는 당신이 하나의 제품을 사용할 수 있습니다. 더 나은 방법 :

if (!obj.isPresent()) {
    logger.fatal(...);   
} else {
    obj.get().setAvailable(true);
}
return obj;

여러 문장으로 분할해야합니다. 수행하는 한 가지 방법은 다음을 수행 할 수 있습니다.

if (!obj.isPresent()) {
  logger.fatal("Object not available");
}

obj.ifPresent(o -> o.setAvailable(true));
return obj;

또 다른 방법 (과도하게 설계 수 있음)은 다음을 사용하는 것입니다 map.

if (!obj.isPresent()) {
  logger.fatal("Object not available");
}

return obj.map(o -> {o.setAvailable(true); return o;});

만약 obj.setAvailable당신이 반환하게 obj, 다음 수 두 번째 예 :

if (!obj.isPresent()) {
  logger.fatal("Object not available");
}

return obj.map(o -> o.setAvailable(true));

우선, dao.find()를 반환 Optional<Obj>하거나 생성해야합니다.

예 :

Optional<Obj> = dao.find();

또는 다음과 같이 스스로 할 수 있습니다.

Optional<Obj> = Optional.ofNullable(dao.find());

이것은 Optional<Obj>존재하거나 Optional.empty()존재하지 않으면 반환 됩니다 .

이제 해결을 찾아 보겠습니다.

public Obj getObjectFromDB() {
   return Optional.ofNullable(dao.find()).flatMap(ob -> {
            ob.setAvailable(true);
            return Optional.of(ob);    
        }).orElseGet(() -> {
            logger.fatal("Object not available");
            return null;
        });
    }

이 당신이 찾고있는 하나의 라이너입니다 :)


Java 8 Spring은 "옵션과 함께 작동하는 메소드"에서 제공 하여 원하는 것을 달성합니다. 예는 다음과 가변적입니다.ifPresentOrElse

import static org.springframework.data.util.Optionals.ifPresentOrElse;    

ifPresentOrElse(dao.find(), obj -> obj.setAvailable(true), () -> logger.fatal("Object not available"));

이다.orElseRun방법은 준비라고 .orElseGet문제 즉, 달리 .map,, .isPresent를 반환하지 않습니다 Optional<Obj>.

하나의 문장에서 아버지가 공연하고 다음이 가능합니다.

public Obj getObjectFromDB() {
    return dao.find()
        .map( obj -> { 
            obj.setAvailable(true);
            return Optional.of(obj); 
         })
        .orElseGet( () -> {
            logger.fatal("Object not available"); 
            return Optional.empty();
    });
}

그러나 이것은 이전에 가졌던 것보다 훨씬 어색합니다.


예를 들어 다음과 같은 몇 가지 "한 줄"솔루션을 만들 수 있습니다.

    obj.map(o -> (Runnable) () -> o.setAvailable(true))
       .orElse(() -> logger.fatal("Object not available"))
       .run();

또는

    obj.map(o -> (Consumer<Object>) c -> o.setAvailable(true))
       .orElse(o -> logger.fatal("Object not available"))
       .accept(null);

또는

    obj.map(o -> (Supplier<Object>) () -> {
            o.setAvailable(true);
            return null;
    }).orElse(() () -> {
            logger.fatal("Object not available")
            return null;
    }).get();

처럼, 뭔가를 아주 허용 좋은 보이지 않는 orElseRun더 나은 것,하지만 난 당신이 정말 한 줄 솔루션을 원하는 경우의 Runnable에 해당 옵션이 생각합니다.


당신은 필요합니다 Optional.isPresent ()OrElse라는 ()을 . 스 니펫은 존재하지 않습니다.

선택적의 요점은 메서드에서 반환하는 것입니다.


dao.find()인스턴스를 반환의하는 메서드를 변경할 수 없다고 가정 Optional<Obj>하므로 적절한 인스턴스를 직접 만들어야합니다.

다음 코드가 도움이 될 것입니다. OptionalActionif-else는 제공 하는 클래스를 만들었습니다 .

public class OptionalTest
{
  public static Optional<DbObject> getObjectFromDb()
  {
    // doa.find()
    DbObject v = find();

    // create appropriate Optional
    Optional<DbObject> object = Optional.ofNullable(v);

    // @formatter:off
    OptionalAction.
    ifPresent(object)
    .then(o -> o.setAvailable(true))
    .elseDo(o -> System.out.println("Fatal! Object not available!"));
    // @formatter:on
    return object;
  }

  public static void main(String[] args)
  {
    Optional<DbObject> object = getObjectFromDb();
    if (object.isPresent())
      System.out.println(object.get());
    else
      System.out.println("There is no object!");
  }

  // find may return null
  public static DbObject find()
  {
    return (Math.random() > 0.5) ? null : new DbObject();
  }

  static class DbObject
  {
    private boolean available = false;

    public boolean isAvailable()
    {
      return available;
    }

    public void setAvailable(boolean available)
    {
      this.available = available;
    }

    @Override
    public String toString()
    {
      return "DbObject [available=" + available + "]";
    }
  }

  static class OptionalAction
  {
    public static <T> IfAction<T> ifPresent(Optional<T> optional)
    {
      return new IfAction<>(optional);
    }

    private static class IfAction<T>
    {
      private final Optional<T> optional;

      public IfAction(Optional<T> optional)
      {
        this.optional = optional;
      }

      public ElseAction<T> then(Consumer<? super T> consumer)
      {
        if (optional.isPresent())
          consumer.accept(optional.get());
        return new ElseAction<>(optional);
      }
    }

    private static class ElseAction<T>
    {
      private final Optional<T> optional;

      public ElseAction(Optional<T> optional)
      {
        this.optional = optional;
      }

      public void elseDo(Consumer<? super T> consumer)
      {
        if (!optional.isPresent())
          consumer.accept(null);
      }
    }
  }
}

참고 URL : https://stackoverflow.com/questions/29235982/how-to-execute-logic-on-optional-if-not-present

반응형