什么会导致 Retrofit 忽略方法调用?
What could cause Retrofit to ignore method invocation?
我已经 运行 变成了 st运行ge 行为。我的应用程序有一个主要 activity,它由 navdrawer 布局和默认列表片段组成。列表片段使用 Retrofit 客户端来获取内容。选择列表项时会显示详细视图。在小屏幕上,详细视图通过单独的 activity 加载。 activity 还使用 Retrofit 客户端来获取详细信息。然后,可以从生成新 activity 的导航抽屉中选择一个选项。那activity也是用Retrofit客户端发数据的
列表和详细信息活动可以很好地调用 Retrofit 方法。我可以看到日志,一切都按预期工作。但是,通过导航选项打开的第三个 activity 尝试调用被 Retrofit 忽略的 Retrofit 客户端方法。
在所有实例中,Retrofit 客户端都以相同的方式实例化:
protected ServerApi buildApi(String serverEndpoint) {
Gson gson = new GsonBuilder()
.create();
RestAdapter restAdapter = new RestAdapter.Builder()
.setLog(new AndroidLog("ServerApi"))
.setLogLevel(RestAdapter.LogLevel.FULL)
.setEndpoint(serverEndpoint)
.setErrorHandler(new ApiErrorHandler())
.setConverter(new GsonConverter(gson))
.build();
return restAdapter.create(ServerApi.class);
}
在列表和详细活动中,我可以看到详细的 Retrofit 日志条目。但是,当我在第三个 activity 中调用类似构造的 Retrofit 客户端时,没有日志输出,也没有抛出错误,logcat 不打印任何内容。它几乎就像调用被忽略了。
我确定服务器端点已正确设置,并且我还确保调用已执行。起初,我认为这可能是由于客户端的创建发生在 onCreate
并在单独的线程上执行。但是,在创建客户端后立即进行测试调用不会产生任何结果。
有谁知道是什么导致了一个 activity 而不是其他人的这种奇怪行为?
更新 1
是的,我试过到处记录。我更绝望的 Retrofit 客户端构建如下所示:
protected class ApiErrorHandler implements ErrorHandler {
@Override
public Throwable handleError(RetrofitError cause) {
Log.e(TAG, "Got API error " + cause);
Response r = cause.getResponse();
Throwable throwable = null;
if (r == null) {
return cause;
}
// Try to interpret the response
String responseBody = new String(((TypedByteArray) r.getBody()).getBytes());
Gson gson = new GsonBuilder().create();
BaseResponse response = null;
try {
response = gson.fromJson(responseBody, BaseResponse.class);
}
catch (Exception e) {
Log.e(TAG, "Could not parse the error response:" + e.getMessage());
}
return new ServerApiException(response.getErrorMessage());
}
}
protected class ApiRequestInterceptor implements RequestInterceptor {
@Override
public void intercept(RequestFacade request) {
Log.d(TAG, "Intercepted API request: " + request.toString());
}
}
RestAdapter restAdapter = new RestAdapter.Builder()
.setLog(new AndroidLog("ServerApi"))
.setLogLevel(RestAdapter.LogLevel.FULL)
.setEndpoint(mServerUrl)
.setErrorHandler(new ApiErrorHandler())
.setConverter(new GsonConverter(gson))
.setRequestInterceptor(new ApiRequestInterceptor())
.setProfiler(new Profiler() {
@Override
public Object beforeCall() {
Log.d("RetrofitProfiler", "Before call to retrofit");
return null;
}
@Override
public void afterCall(Profiler.RequestInformation requestInfo, long elapsedTime, int statusCode, Object beforeCallData) {
Log.d("RetrofitProfiler", String.format("HTTP %d %s %s (%dms)",
statusCode, requestInfo.getMethod(), requestInfo.getRelativePath(), elapsedTime));
}
})
.build();
更新 2
好吧,我错了,但我不知道 Retrofit 至少需要一个订阅者在场。我想这是有道理的,现在我想到了……但它不是一劳永逸的。
我做错了什么:
mClient = new ServerClient(mServerEndpoint, mAccountToken);
mUvClient.getFeed(mFeedId);
这是我不知道的部分。您需要订阅可观察对象才能执行:
mClient = new ServerClient(mServerEndpoint, mAccountToken);
mUvClient.getFeed(mFeedId)
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<FeedInfo>() {
@Override
public void onNext(FeedInfo feedInfo) {
Log.d(TAG, "on Next");
}
@Override
public void onCompleted() {
Log.d(TAG, "on completed");
}
@Override
public void onError(Throwable e) {
Log.d(TAG, "on error");
}
});
这会产生预期的 Retrofit 详细日志和结果。无论如何,我希望这对某人有所帮助。
问题是我没有订阅 Retrofit API 生成的 Observable。我以为它无论如何都会开火,但它没有开火,需要订阅。希望这对可能遇到此问题的其他人有所帮助。
我已经 运行 变成了 st运行ge 行为。我的应用程序有一个主要 activity,它由 navdrawer 布局和默认列表片段组成。列表片段使用 Retrofit 客户端来获取内容。选择列表项时会显示详细视图。在小屏幕上,详细视图通过单独的 activity 加载。 activity 还使用 Retrofit 客户端来获取详细信息。然后,可以从生成新 activity 的导航抽屉中选择一个选项。那activity也是用Retrofit客户端发数据的
列表和详细信息活动可以很好地调用 Retrofit 方法。我可以看到日志,一切都按预期工作。但是,通过导航选项打开的第三个 activity 尝试调用被 Retrofit 忽略的 Retrofit 客户端方法。
在所有实例中,Retrofit 客户端都以相同的方式实例化:
protected ServerApi buildApi(String serverEndpoint) {
Gson gson = new GsonBuilder()
.create();
RestAdapter restAdapter = new RestAdapter.Builder()
.setLog(new AndroidLog("ServerApi"))
.setLogLevel(RestAdapter.LogLevel.FULL)
.setEndpoint(serverEndpoint)
.setErrorHandler(new ApiErrorHandler())
.setConverter(new GsonConverter(gson))
.build();
return restAdapter.create(ServerApi.class);
}
在列表和详细活动中,我可以看到详细的 Retrofit 日志条目。但是,当我在第三个 activity 中调用类似构造的 Retrofit 客户端时,没有日志输出,也没有抛出错误,logcat 不打印任何内容。它几乎就像调用被忽略了。
我确定服务器端点已正确设置,并且我还确保调用已执行。起初,我认为这可能是由于客户端的创建发生在 onCreate
并在单独的线程上执行。但是,在创建客户端后立即进行测试调用不会产生任何结果。
有谁知道是什么导致了一个 activity 而不是其他人的这种奇怪行为?
更新 1
是的,我试过到处记录。我更绝望的 Retrofit 客户端构建如下所示:
protected class ApiErrorHandler implements ErrorHandler {
@Override
public Throwable handleError(RetrofitError cause) {
Log.e(TAG, "Got API error " + cause);
Response r = cause.getResponse();
Throwable throwable = null;
if (r == null) {
return cause;
}
// Try to interpret the response
String responseBody = new String(((TypedByteArray) r.getBody()).getBytes());
Gson gson = new GsonBuilder().create();
BaseResponse response = null;
try {
response = gson.fromJson(responseBody, BaseResponse.class);
}
catch (Exception e) {
Log.e(TAG, "Could not parse the error response:" + e.getMessage());
}
return new ServerApiException(response.getErrorMessage());
}
}
protected class ApiRequestInterceptor implements RequestInterceptor {
@Override
public void intercept(RequestFacade request) {
Log.d(TAG, "Intercepted API request: " + request.toString());
}
}
RestAdapter restAdapter = new RestAdapter.Builder()
.setLog(new AndroidLog("ServerApi"))
.setLogLevel(RestAdapter.LogLevel.FULL)
.setEndpoint(mServerUrl)
.setErrorHandler(new ApiErrorHandler())
.setConverter(new GsonConverter(gson))
.setRequestInterceptor(new ApiRequestInterceptor())
.setProfiler(new Profiler() {
@Override
public Object beforeCall() {
Log.d("RetrofitProfiler", "Before call to retrofit");
return null;
}
@Override
public void afterCall(Profiler.RequestInformation requestInfo, long elapsedTime, int statusCode, Object beforeCallData) {
Log.d("RetrofitProfiler", String.format("HTTP %d %s %s (%dms)",
statusCode, requestInfo.getMethod(), requestInfo.getRelativePath(), elapsedTime));
}
})
.build();
更新 2
好吧,我错了,但我不知道 Retrofit 至少需要一个订阅者在场。我想这是有道理的,现在我想到了……但它不是一劳永逸的。
我做错了什么:
mClient = new ServerClient(mServerEndpoint, mAccountToken);
mUvClient.getFeed(mFeedId);
这是我不知道的部分。您需要订阅可观察对象才能执行:
mClient = new ServerClient(mServerEndpoint, mAccountToken);
mUvClient.getFeed(mFeedId)
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<FeedInfo>() {
@Override
public void onNext(FeedInfo feedInfo) {
Log.d(TAG, "on Next");
}
@Override
public void onCompleted() {
Log.d(TAG, "on completed");
}
@Override
public void onError(Throwable e) {
Log.d(TAG, "on error");
}
});
这会产生预期的 Retrofit 详细日志和结果。无论如何,我希望这对某人有所帮助。
问题是我没有订阅 Retrofit API 生成的 Observable。我以为它无论如何都会开火,但它没有开火,需要订阅。希望这对可能遇到此问题的其他人有所帮助。