Java 使用 ReactiveX 进行 Http 调用
Make Http call using ReactiveX for Java
我是 Java 的 ReactiveX 新手,我有以下代码块可以进行外部 http 调用,但它不是异步的。我们使用的是 rxjava 1.2,以及 Java 1.8
private ResponseEntity<String> callExternalUrl(String url, String json, HttpMethod method) {
RestTemplate restTemplate;
HttpEntity request;
request = new HttpEntity(jsonContent, httpHeaders);
return restTemplate.exchange(url, httpMethod, request, String.class);
}
我在网上找到了以下代码块,但我无法完全理解它以及如何将它应用到我的代码库中。
private RxClient<RxObservableInvoker> httpClient;
public <T> Observable<T> fetchResult(String url, Func1<Response, T> mapper) {
return httpClient.target(url)
.request()
.rx()
.get()
.subscribeOn(Schedulers.io())
.map(mapper);
}
如果我理解正确的话,你需要这样的东西来包装你现有的 callExternalUrl
static Observable<String> callExternalUrlAsync(String url, String json, HttpMethod method)
{
return Observable.fromCallable(() -> callExternalUrl(url, json, method))
.subscribeOn(Schedulers.io())
.flatMap(re -> {
if (re.hasBody())
return Observable.just(re.getBody());
else
return Observable.error(new RuntimeException("Bad response status " + re.getStatusCode()));
},
e -> Observable.error(e),
(Func0<Observable<? extends String>>) (() -> Observable.empty())) // I need explicit cast or it won't compile :-(
.observeOn(Schedulers.computation());
}
代码的简短说明:
- 它计划在
Schedulers.io
上执行现有的 callExternalUrl
- 将
ResponseEntity<T>
最小化为成功的 T
和错误案例。它也发生在 io
调度程序上,但并不重要,因为它真的很短。 (如果 callExternalUrl
中有异常,则按原样传递。)
- 订阅要在
Schedulers.computation
上执行的结果
注意事项:
- 您可能想为
subscribeOn
和 observeOn
使用自定义调度程序
- 您可能希望在传递给
flatMap
的第一个 lambda 中有一些更好的逻辑来区分成功和错误,并且您肯定想要一些更具体的异常类型。
高阶魔法
如果您愿意使用高阶函数并牺牲一点性能来减少代码重复,您可以这样做:
// Universal wrapper method
static <T> Observable<T> wrapCallExternalAsAsync(Func3<String, String, HttpMethod, ResponseEntity<T>> externalCall, String url, String json, HttpMethod method)
{
return Observable.fromCallable(() -> externalCall.call(url, json, method))
.subscribeOn(Schedulers.io())
.flatMap(re -> {
if (re.hasBody())
return Observable.just(re.getBody());
else
return Observable.error(new RuntimeException("Bad response status " + re.getStatusCode()));
},
e -> Observable.error(e),
(Func0<Observable<? extends T>>) (() -> Observable.empty())) // I need explicit cast or it won't compile :-(
.observeOn(Schedulers.computation());
}
static Observable<String> callExternalUrlAsync_HigherOrder(String url, String json, HttpMethod method)
{
return wrapCallExternalAsAsync(MyClass::callExternalUrl, url, json, method);
}
MyClass
就是你的 callExternalUrl
所在的地方。
更新(仅限异步调用)
private static RxClient httpClient = Rx.newClient(RxObservableInvoker.class); // 在这里你可以传递自定义的 ExecutorService
private <T> Observable<String> executeHttpAsync(String url, String httpMethod, Entity<T> entity) {
return httpClient.target(url)
.request()
.headers(httpHeaders) // assuming httpHeaders is something global as in your example
.rx()
.method(httpMethod, entity)
.map(resp -> {
if (200 != resp.getStatus()) {
throw new RuntimeException("Bad status code " + resp.getStatus());
} else {
if (!resp.hasEntity()) {
// return null; // or error?
throw new RuntimeException("Empty response"); // or empty?
} else {
try {
return resp.readEntity(String.class);
} catch (Exception ex) {
throw new RuntimeException(ex); // wrap exception into unchecked
}
}
}
})
.observeOn(Schedulers.computation());
}
private Observable<String> executeGetAsync(String url) {
return executeHttpAsync(url, "GET", null);
}
private Observable<String> executePostAsync(String url, String json) {
return executeHttpAsync(url, "POST", Entity.json(json));
}
再次类似注意事项适用:
- 您可能希望对
newClient
调用和 observeOn
使用自定义调度程序
- 您可能希望有一些更好的错误处理逻辑,而不仅仅是检查它是否为 HTTP 200,并且您肯定需要一些更具体的异常类型。但这都是特定于业务逻辑的,因此由您决定。
此外,从您的示例中也不清楚请求的主体 (HttpEntity
) 是如何构建的,以及您是否真的总是希望 String
作为响应,就像在您的原始示例中一样.不过我只是照原样复制了你的逻辑。如果您需要更多信息,您可能应该参考 https://jersey.java.net/documentation/2.25/media.html#json
上的文档
我是 Java 的 ReactiveX 新手,我有以下代码块可以进行外部 http 调用,但它不是异步的。我们使用的是 rxjava 1.2,以及 Java 1.8
private ResponseEntity<String> callExternalUrl(String url, String json, HttpMethod method) {
RestTemplate restTemplate;
HttpEntity request;
request = new HttpEntity(jsonContent, httpHeaders);
return restTemplate.exchange(url, httpMethod, request, String.class);
}
我在网上找到了以下代码块,但我无法完全理解它以及如何将它应用到我的代码库中。
private RxClient<RxObservableInvoker> httpClient;
public <T> Observable<T> fetchResult(String url, Func1<Response, T> mapper) {
return httpClient.target(url)
.request()
.rx()
.get()
.subscribeOn(Schedulers.io())
.map(mapper);
}
如果我理解正确的话,你需要这样的东西来包装你现有的 callExternalUrl
static Observable<String> callExternalUrlAsync(String url, String json, HttpMethod method)
{
return Observable.fromCallable(() -> callExternalUrl(url, json, method))
.subscribeOn(Schedulers.io())
.flatMap(re -> {
if (re.hasBody())
return Observable.just(re.getBody());
else
return Observable.error(new RuntimeException("Bad response status " + re.getStatusCode()));
},
e -> Observable.error(e),
(Func0<Observable<? extends String>>) (() -> Observable.empty())) // I need explicit cast or it won't compile :-(
.observeOn(Schedulers.computation());
}
代码的简短说明:
- 它计划在
Schedulers.io
上执行现有的 - 将
ResponseEntity<T>
最小化为成功的T
和错误案例。它也发生在io
调度程序上,但并不重要,因为它真的很短。 (如果callExternalUrl
中有异常,则按原样传递。) - 订阅要在
Schedulers.computation
上执行的结果
callExternalUrl
注意事项:
- 您可能想为
subscribeOn
和observeOn
使用自定义调度程序
- 您可能希望在传递给
flatMap
的第一个 lambda 中有一些更好的逻辑来区分成功和错误,并且您肯定想要一些更具体的异常类型。
高阶魔法
如果您愿意使用高阶函数并牺牲一点性能来减少代码重复,您可以这样做:
// Universal wrapper method
static <T> Observable<T> wrapCallExternalAsAsync(Func3<String, String, HttpMethod, ResponseEntity<T>> externalCall, String url, String json, HttpMethod method)
{
return Observable.fromCallable(() -> externalCall.call(url, json, method))
.subscribeOn(Schedulers.io())
.flatMap(re -> {
if (re.hasBody())
return Observable.just(re.getBody());
else
return Observable.error(new RuntimeException("Bad response status " + re.getStatusCode()));
},
e -> Observable.error(e),
(Func0<Observable<? extends T>>) (() -> Observable.empty())) // I need explicit cast or it won't compile :-(
.observeOn(Schedulers.computation());
}
static Observable<String> callExternalUrlAsync_HigherOrder(String url, String json, HttpMethod method)
{
return wrapCallExternalAsAsync(MyClass::callExternalUrl, url, json, method);
}
MyClass
就是你的 callExternalUrl
所在的地方。
更新(仅限异步调用)
private static RxClient httpClient = Rx.newClient(RxObservableInvoker.class); // 在这里你可以传递自定义的 ExecutorService
private <T> Observable<String> executeHttpAsync(String url, String httpMethod, Entity<T> entity) {
return httpClient.target(url)
.request()
.headers(httpHeaders) // assuming httpHeaders is something global as in your example
.rx()
.method(httpMethod, entity)
.map(resp -> {
if (200 != resp.getStatus()) {
throw new RuntimeException("Bad status code " + resp.getStatus());
} else {
if (!resp.hasEntity()) {
// return null; // or error?
throw new RuntimeException("Empty response"); // or empty?
} else {
try {
return resp.readEntity(String.class);
} catch (Exception ex) {
throw new RuntimeException(ex); // wrap exception into unchecked
}
}
}
})
.observeOn(Schedulers.computation());
}
private Observable<String> executeGetAsync(String url) {
return executeHttpAsync(url, "GET", null);
}
private Observable<String> executePostAsync(String url, String json) {
return executeHttpAsync(url, "POST", Entity.json(json));
}
再次类似注意事项适用:
- 您可能希望对
newClient
调用和observeOn
使用自定义调度程序
- 您可能希望有一些更好的错误处理逻辑,而不仅仅是检查它是否为 HTTP 200,并且您肯定需要一些更具体的异常类型。但这都是特定于业务逻辑的,因此由您决定。
此外,从您的示例中也不清楚请求的主体 (HttpEntity
) 是如何构建的,以及您是否真的总是希望 String
作为响应,就像在您的原始示例中一样.不过我只是照原样复制了你的逻辑。如果您需要更多信息,您可能应该参考 https://jersey.java.net/documentation/2.25/media.html#json