1

我在使用 EventBus 3.0.0 时遇到问题,我在其中发布了一个这样的事件:

Call<List<SessionSpec>> call = httpService.getSessionSpecs();
call.enqueue(new Callback<List<SessionSpec>>() {

   @Override
   public void onResponse(Call<List<SessionSpec>> call, Response<List<SessionSpec>> response)  {
       if (response.isSuccessful())  {
           List<SessionSpec> specs = response.body();
           EventBus.getDefault().post((List<SessionSpec>)specs);
       }
       else Log.e(TAG, "sendSessionSpecs(): request NOT successful");
   }

   @Override
   public void onFailure(Call<List<SessionSpec>> call, Throwable t)  {
      Log.e(TAG, "sendSessionsSpecs(): failed");
   }
});

我在同一个片段中有两个订阅者,每个订阅者都有不同的签名,但他们都从一个帖子中被调用:

@Subscribe
public void onSessionSpec(List<SessionSpec> specs) {
    Log.d(TAG, "onSessionSpec(): entered");
    Log.d(TAG, "    : number of session specs: " + specs.size());
}

第二个订阅者定义为:

@Subscribe
public void onOverlayType(List<OverlayType> types) {
    Log.d(TAG, "onOverlayType(): entered");
    Log.d(TAG, "    : number of overlay types: " + types.size());
}

这两个回调都在一个片段中,该片段在帖子完成时处于活动状态,并且我已验证帖子仅被调用一次。当发布单个 SessionSpec 事件时,onSessionSpec 和 onOverlayType 回调都由 EventBus 调度,事件类型为 List>,因此 onOverlayType 回调在其回调参数中接收到错误的类型。OverlayType 类是一个简单的 POJO 类,它有 2 个成员,一个 int "sid" 和一个 String "name"。SessionSpec 类更复杂;它确实有一个成员字符串“名称”,但除此之外,这两个类之间没有其他共同点。我已经验证在 SessionSpec 类中没有与“OverlayType”非常相似的东西。

接口定义是这样的:

public interface VcapHttpInterface {

    @GET("overlay/types")
    Call<List<OverlayType>> getOverlayTypes();

    @GET("session/list")
    Call<List<SessionSpec>> getSessionSpecs();

    @GET("session/{id}")
    Call<Session> getSession(@Path("id") int sid);

}

getSession 事件发布/回调没有问题。

我整天都在试图弄清楚出了什么问题,所以我现在一无所知。有人知道我的代码可能有什么问题吗?

谢谢,-安德烈斯

编辑:EventBus 如何知道为特定响应调用哪个处理程序?我读过的一些帖子说 EventBus 不使用处理程序签名,但它怎么知道如何将响应映射到正确的订阅处理程序例程?有没有办法为给定事件显式定义处理程序回调?

4

2 回答 2

2

EventBus检查您发布的对象的类,并在其参数中调用期望该类的方法。在您的情况下,您发布的对象是List. 在您的两个听众中,您都希望有一个 type 的对象List。不管你输入什么泛型OverlayTypeSessionSpeceventbus 都会调用两者。为了使其工作,您必须将模型定义为事件。

public class OverlayTypeEvent {

    public List<OverlayType> types;

    public OverlayTypeEvent(List<OverlayType> types) {
        this.types = types;
    }
}

public class SessionSpecEvent {

    public List<SessionSpec> types;

    public SessionSpecEvent(List<SessionSpec> types) {
        this.types = types;
    }
}

并单独听他们说。然后发布具有特定类型的事件。

@Subscribe
public void onSessionSpec(OverlayTypeEvent event) {
    List<OverlayType> overlayTypes = event.overlayType;
}
于 2017-03-12T02:08:34.990 回答
0

如果您不想在每次发送列表数据时都将新类创建为容器,您可以将其配对为简单容器,它有两个通用字段(第一个第二个)来包含变量。

您可以使用first作为 key 来检查类的类型,second包含实际数据。

List<SessionSpec> specs = response.body();
EventBus.getDefault().post(new Pair<>(SessionSpec.class.getSimpleName(), specs));

接收数据:

@Subscribe
public void onSessionSpec(Pair<String, List<SessionSpec>> specContainer){
    if (SessionSpec.class.getSimpleName().equals(specContainer.first)) {
        List<SessionSpec> sessionSpecs = specContainer.second;
    }
}

@Subscribe
public void onOverlayType(Pair<String, List<OverlayType>> overlayContainer) {
    if (OverlayType.class.getSimpleName().equals(overlayContainer.first)) {
        List<OverlayType> overlayTypes = overlayContainer.second;
    }
}

此解决方案的优点:减少创建不需要的类。
缺点onSessionSpeconOverlayType都被调用。

于 2018-10-01T05:05:35.830 回答