我正在使用OkHttp客户端和Jackson进行Json Serialization的Retrofit,并希望得到响应的标题。
我知道我可以扩展OkClient 并拦截它。但这是在反序列化过程开始之前。
我基本上需要的是将标题与反序列化的Json对象一起使用。
答案 0 :(得分:39)
使用Retrofit 1.9.0,如果使用接口的Callback异步版本,
@GET("/user")
void getUser(Callback<User> callback)
然后你的回调会收到一个Response
对象
Callback<User> user = new Callback<User>() {
@Override
public void success(User user, Response response) {
}
@Override
public void failure(RetrofitError error) {
}
}
其中有一个名为getHeaders()
Callback<User> user = new Callback<User>() {
@Override
public void success(User user, Response response) {
List<Header> headerList = response.getHeaders();
for(Header header : headerList) {
Log.d(TAG, header.getName() + " " + header.getValue());
}
}
对于Retrofit 2.0的界面,您可以使用Call<T>
执行此操作。
对于Retrofit 2.0的Rx支持,您可以使用Observable<Result<T>>
答案 1 :(得分:32)
在Retrofit 2.0.0中,您可以获得如下标题:
public interface Api {
@GET("user")
Call<User> getUser();
}
Call<User> call = api.getUser();
call.enqueue(new Callback<User>() {
@Override
public void onResponse(Call<User> call, Response<User> response) {
// get headers
Headers headers = response.headers();
// get header value
String cookie = response.headers().get("Set-Cookie");
// TODO
}
@Override
public void onFailure(Call<User> call, Throwable t) {
// TODO
}
});
答案 2 :(得分:2)
就像你一样,我想要有效载荷旁边的标题。我需要访问Etag。它需要一些复古,但你可以做到。这就是我的所作所为。这是一个肮脏的样本,所以不要把它作为最佳实践样本。
public static RestAdapter.Builder getRestBuilder(Context context) {
GsonBuilder gsonBuilder = GsonBuilderUtils.getBuilder();
Gson gson = gsonBuilder.create();
// **
// 1. create our own custom deserializer here
// **
final MyGsonConverter gsonConverter = new MyGsonConverter(gson);
OkHttpClient httpClient = MyPersonalOkHttpFactory.getInstance().getAuthHttpClient(context);
httpClient.networkInterceptors().add(new Interceptor() {
@Override
public Response intercept(Chain chain) throws IOException {
Request originalRequest = chain.request();
Response response = chain.proceed(originalRequest);
// **
// 2. add the headers from the Interceptor to our deserializer instance
// **
gsonConverter.headers = response.headers();
return response;
}
});
RestAdapter.Builder builder = new RestAdapter.Builder()
.setClient(new OkClient(httpClient))
.setEndpoint(Common.getApiOriginUrl())
.setConverter(gsonConverter);
return builder;
}
private static class MyGsonConverter extends GsonConverter {
private Headers headers;
public MyGsonConverter(Gson gson) {
super(gson);
}
@Override
public Object fromBody(TypedInput body, Type type) throws ConversionException {
Object obj = super.fromBody(body, type);
// **
// 3. at this point, gson is called and you have access to headers
// do whatever you want here. I just set it on the return object.
// **
if (obj instanceof HeadersArrayList) {
((HeadersArrayList)obj).setHeaders(headers);
}
return obj;
}
}
public class HeadersArrayList<K> extends ArrayList<K>{
private Headers headers;
public Headers getHeaders() {
return headers;
}
public void setHeaders(Headers headers) {
this.headers = headers;
}
}
// the retrofit api for reference
@GET("/api/of/my/backend/{stuff}")
HeadersArrayList<String> getSomething(@Path("stuff") String stuff);
答案 3 :(得分:0)
首先打印整个响应,正文,代码,消息,标题(通过记录或其他方式),然后尝试从中找到线索。
我建议您阅读API文档,并查看请求的类型。
使用邮递员检查以下哪一项有效: 1.表格数据 2.x-www-form-Urlencoded 3.原始 4.binary
然后相应地在接口的方法声明中设置注释。
例如:在我的情况下,它采用的是x-www-form-Urlencoded,因此我不得不使用
@FormUrlEncoded @Headers(“ Content-Type:application / x-www-form-urlencoded”)
在方法声明中。
然后将@Field注释用于我发送的单个值 像
Call <'ReturnObj'> Signup(@Field(“ name”)字符串名称,@Field(“ phoneNumber”)长phoneNumber,@Field(“ password”)字符串密码,@Field(“ counter”)int计数器);