RxJava + Retrofit 2 链接 API 请求

问题描述 投票:0回答:3

我需要通过将多个改造 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 列表。

android rx-java retrofit2 dagger-2
3个回答
3
投票

为了实现您的目标,您应该遵循下一个算法:

  1. 向服务器请求项目列表。现在你的 observable 会发出
    List<Item>
    ;
  2. List<Item>
    的发射变换为仅发射
    Item
  3. 对于
    Item
    的每个实例,您应该请求
    ItemDetail
    的实例。由于
    mItemDetailsService.getItemDetails(@Path("id") int id)
    返回
    Observable<ItemDetail>
    ,因此您应该使用
    flatMap(...)
    来实现此目的;
  4. 合并列表中的 ItemDetail 响应。

代码:

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();
        }
     })

0
投票

如果我理解正确,那么这应该可以满足您的需求:

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,以使其更清楚我的意思。


0
投票
getItems().flatMap(
    // ...
    return Observable.from(list);
    // ...
).flatMap(
    // ...
    getItemDetails(id)
    // ...
).toList();
© www.soinside.com 2019 - 2024. All rights reserved.