我需要通过将多个改造 API 调用链接在一起来创建一个可观察对象。
我有 2 个服务:ItemService 仅包含项目名称和 id,ItemDetailsService 包含有关项目的许多其他详细信息,如描述图像等。我正在使用 Retrofit 2 + RxJava + Dagger 2。
项目服务:
@GET("apione/items")
Observable<List<Items>> getItems();
商品详情服务:
@GET("apitwo/item/{id}")
Observable<ItemDetail> getItemDetails(@Path("id") int id);
项目 JSON:
[
{
"id": 1,
"name": "one"
},
{
"id": 2,
"name": "two"
},
{
"id": 3,
"name": "three"
}
]
具体项目 id 的 ItemDetails JSON:
{
"id": 1,
"image_url": "http://.../1.png",
"description": "description of item one",
"category": "cat_1",
"quantity" 10
}
这 2 个 API 也有不同的基本 URL。那么如何才能创建一个 Observable 返回包含图像、数量和类别的项目列表呢?
已更新
我被困在这里了。这是我的ItemPresenter
public class ItemPresenter extends BasePresenter<ItemView> implements Observer<List<Item>> {
@Inject protected ItemService mItemService;
@Inject protected ItemDetailsService mItemDetailsService;
@Inject
public ItemPresenter() {
}
public void getItems() {
getView().onShowDialog("Loading");
Observable<List<Item>> itemObservables= mItemService.getItems()
.flatMap(new Func1<List<Item>, Observable<ItemDetail>>() {
@Override
public Observable<ItemDetail> call(List<Item> items) {
List<Observable<ItemDetail>> detailObservables = new ArrayList<>();
for (Item item : items) {
detailObservables.add(mItemDetailsService.getItemDetail(item.getId());
}
return Observable.zip(
detailObservables,
args -> {
List<Item> itemDetails = new ArrayList<>();
for (Object arg : args) {
itemDetails.add((Item) arg);
}
return itemDetails;
}
);
}
});
subscribe(detailObservables, this);
}
@Override
public void onCompleted() {
getView().onHideDialog();
getView().onShowToast("Complete");
}
@Override
public void onError(Throwable e) {
getView().onHideDialog();
getView().onShowToast("Error loading " + e.getMessage());
}
@Override
public void onNext(List<Item> items) {
getView().onItems(items);
}
}
不明白如何正确制作。如果我只需要列表中项目的一些详细信息,而不是 ItemDetail 中的所有内容,我该怎么办?
我只需要从 ItemDetail 添加一些详细信息到 Items 列表。
为了实现您的目标,您应该遵循下一个算法:
List<Item>
;List<Item>
的发射变换为仅发射 Item
;Item
的每个实例,您应该请求 ItemDetail
的实例。由于 mItemDetailsService.getItemDetails(@Path("id") int id)
返回 Observable<ItemDetail>
,因此您应该使用 flatMap(...)
来实现此目的;代码:
Observable<List<ItemDetail>> detailObservables = mItemService.getItems()
.flatMap(new Func1<List<Item>, Observable<Item>>() {
@Override
public Observable<Item> call(List<Item> items) {
return Observable.from(items);
}
})
.flatMap(new Func1<Item, Observable<ItemDetail>>() {
@Override
public Observable<ItemDetail> call(Item item) {
return mItemDetailsService.getItemDetail(item.getId());
}
}).toList();
例如,如果您只需要 image_url 而不是整个项目详细信息,您可以致电:
.map(new Func1<ItemDetail, String>() {
@Override
public String call(Object itemDetail) {
return itemDetail.getImageUrl();
}
})
如果我理解正确,那么这应该可以满足您的需求:
private Observable<List<ItemDetail>> getItemDetails() {
return service1.getItems()
.flatMap(new Func1<List<Item>, Observable<List<ItemDetail>>>() {
@Override
public Observable<List<ItemDetail>> call(List<Item> items) {
List<Observable<ItemDetail>> detailObservables = new ArrayList<>();
for (Item item : items) {
detailObservables.add(service2.getItemDetails(item.id));
}
return Observable.zip(
detailObservables,
new FuncN<List<ItemDetail>>() {
@Override
public List<ItemDetail> call(Object... args) {
List<ItemDetail> itemDetails = new ArrayList<>();
for (Object arg : args) {
itemDetails.add((ItemDetail) arg);
}
return itemDetails;
}
}
);
}
});
}
注意,我在这段代码中假设您的
getItemDetails
方法返回 ItemDetail
对象列表,只是为了代码清晰。
说明: 我使用了
flatMap()
运算符来处理从第一个服务请求接收到的项目列表,然后为每个项目详细信息 API 请求创建一个 Observable 并将其添加到 detailObservables
列表中。然后使用 zip
运算符,它基本上启动所有请求,并在所有请求完成时触发 FuncN
中的回调。您必须将 Object
投射到 ItemDetail
那里。
编辑:我在方法名称前面添加了service1和service2,以使其更清楚我的意思。
getItems().flatMap(
// ...
return Observable.from(list);
// ...
).flatMap(
// ...
getItemDetails(id)
// ...
).toList();