4

我目前正在努力尝试使用 rx 实现 tcp 看门狗/重试系统,非常感谢您的帮助。

有一个 Observable,我希望有一个 Observable,它会定期检查我们是否仍然可以写入套接字。很简单,我可以这样做:

class SocketSubscribeFunc implements Observable.OnSubscribeFunc<Socket> {
  private final String hostname;
  private final int port;
  private Socket socket;

  SocketSubscribeFunc(String hostname, int port) {
    this.hostname = hostname;
    this.port = port;
  }

  public Subscription onSubscribe(final Observer<? super Socket> observer) {
    try {
      log.debug("Trying to connect...");
      socket = new Socket(hostname, port);
      observer.onNext(socket);
    } catch (IOException e) {
      observer.onError(e);
    }
    return new Subscription() {
      public void unsubscribe() {
        try {
          socket.close();
        } catch (IOException e) {
          e.printStackTrace();
        }
      }
    };
  }
}

Observable<Socket> socketObservable = Observable.create(new SocketSubscribeFunc(hostname,port));
Observable<Boolean> watchdog = Observable.combineLatest(socketObservable, Observable.interval(1, TimeUnit.SECONDS), new Func2<Socket, Long, Boolean>() {

  public Boolean call(final Socket socket, final Long aLong) {
    try {
      socket.getOutputStream().write("ping\n".getBytes());
      return true;
    } catch (IOException e) {
     return false;
    }
  }
});

现在,如果可以获取套接字(服务器/链接在创建时关闭)或变得不可写(成功连接后服务器/链接无法访问),我想重试连接。理想情况下,通过重新订阅其 OnSubscribeFunc 使用重试运算符创建连接的套接字 Observable。如您所见,这会在套接字和看门狗 Observables 之间引入循环依赖。我玩弄了一段时间 switchMap/materialize... 为了传播最终的错误无济于事。

我接近放弃这个想法并使用副作用代码中的主题。但是在全球范围内应该有更好的方法:)

提前致谢!

4

1 回答 1

6

首先,我会避免Observable.create大部分时间,因为它通常是不需要的并且会引入不必要的复杂性。在这种情况下,Rx 有一个称为操作符的操作符using,它允许您创建一个在 Observable 的生命周期中存在的资源对象。它自动捕获运行时错误,并且还提供了一个 dispose 操作,因此这对于这个用例中的套接字来说是完美的。我正在使用 Java8 lambda,因为它们更容易伪代码。

Observable.using(
    // Resource (socket) factory
    () -> {
      try {
        return new Socket(hostname, port);
      } catch (IOException e) {
        // Rx will propagate this as an onError event.
        throw new RuntimeException(e);
      }
    },
    // Observable factory
    (socket) -> {
      return Observable.interval(1, TimeUnit.SECONDS)
          .map((unusedTick) {
            try {
              socket.getOutputStream().write("ping\n".getBytes());
              return true;
            } catch (IOException e) {
              throw new RuntimeException(e);
            }
          })
          // Retry the inner job up to 3 times before propagating.
          .retry(3);
    },
    // Dispose action for socket.
    // In real life the close probably needs a try/catch.
    (socket) -> socket.close())
    // Retry the outer job up to 3 times.
    .retry(3)
    // If we propagate all errors, emit a 'false', signaling service is not available.
    .onErrorResumeNext(Observable.just(false));

请注意,如果内部作业传播(在 3 次失败后),这将重试外部作业。要解决此问题,您应该使用谓词以及 retryWhen 来查看有关重试的文档。如果这不是内部作业传播的类型,您可以抛出一个特殊的 RuntimeException 并仅重试外部作业。

using文档:http://reactivex.io/RxJava/javadoc/rx/Observable.html#using(rx.functions.Func0,%20rx.functions.Func1,%20rx.functions.Action1)

于 2015-01-25T23:28:41.847 回答