Я знаю, что эта ветка довольно старая, но у меня была эта проблема, и я придумал классное решение, которое может быть очень полезным для многих, потому что оно исправляет / расширяет библиотеку Volley во многих аспектах.
Я заметил некоторые не поддерживаемые готовые функции Volley:
- Это
JSONObjectRequest
не идеально: вы должны ожидать JSON
в конце (см. Response.Listener<JSONObject>
).
- А что насчет пустых ответов (только со статусом 200)?
- Что мне делать, если я хочу получить свой POJO напрямую из
ResponseListener
?
Я более или менее скомпилировал множество решений в большом универсальном классе, чтобы найти решение для всей проблемы, которую я цитировал.
public class GenericRequest<T> extends JsonRequest<T> {
private final Gson gson = new Gson();
private final Class<T> clazz;
private final Map<String, String> headers;
private boolean muteRequest = false;
private GenericRequest(int method, Class<T> classtype, String url, String requestBody,
Response.Listener<T> listener, Response.ErrorListener errorListener, Map<String, String> headers) {
super(method, url, requestBody, listener,
errorListener);
clazz = classtype;
this.headers = headers;
configureRequest();
}
public GenericRequest(int method, String url, Class<T> classtype, Object toBeSent,
Response.Listener<T> listener, Response.ErrorListener errorListener, Map<String, String> headers) {
this(method, classtype, url, new Gson().toJson(toBeSent), listener,
errorListener, headers);
}
public GenericRequest(int method, String url, Class<T> classtype, Object toBeSent,
Response.Listener<T> listener, Response.ErrorListener errorListener) {
this(method, classtype, url, new Gson().toJson(toBeSent), listener,
errorListener, new HashMap<String, String>());
}
public GenericRequest(int method, String url, Class<T> classtype, String requestBody,
Response.Listener<T> listener, Response.ErrorListener errorListener) {
this(method, classtype, url, requestBody, listener,
errorListener, new HashMap<String, String>());
}
public GenericRequest(String url, Class<T> classtype, Response.Listener<T> listener, Response.ErrorListener errorListener) {
this(Request.Method.GET, url, classtype, "", listener, errorListener);
}
public GenericRequest(String url, Class<T> classtype, Response.Listener<T> listener, Response.ErrorListener errorListener, Map<String, String> headers) {
this(Request.Method.GET, classtype, url, "", listener, errorListener, headers);
}
public GenericRequest(int method, String url, Class<T> classtype, Object toBeSent,
Response.Listener<T> listener, Response.ErrorListener errorListener, Map<String, String> headers, boolean mute) {
this(method, classtype, url, new Gson().toJson(toBeSent), listener,
errorListener, headers);
this.muteRequest = mute;
}
public GenericRequest(int method, String url, Class<T> classtype, Object toBeSent,
Response.Listener<T> listener, Response.ErrorListener errorListener, boolean mute) {
this(method, classtype, url, new Gson().toJson(toBeSent), listener,
errorListener, new HashMap<String, String>());
this.muteRequest = mute;
}
public GenericRequest(int method, String url, Class<T> classtype, String requestBody,
Response.Listener<T> listener, Response.ErrorListener errorListener, boolean mute) {
this(method, classtype, url, requestBody, listener,
errorListener, new HashMap<String, String>());
this.muteRequest = mute;
}
@Override
protected Response<T> parseNetworkResponse(NetworkResponse response) {
if (muteRequest) {
if (response.statusCode >= 200 && response.statusCode <= 299) {
return Response.success(null, HttpHeaderParser.parseCacheHeaders(response));
}
} else {
try {
String json = new String(response.data, HttpHeaderParser.parseCharset(response.headers));
T parsedObject = gson.fromJson(json, clazz);
return Response.success(parsedObject, HttpHeaderParser.parseCacheHeaders(response));
} catch (UnsupportedEncodingException e) {
return Response.error(new ParseError(e));
} catch (JsonSyntaxException e) {
return Response.error(new ParseError(e));
}
}
return null;
}
@Override
public Map<String, String> getHeaders() throws AuthFailureError {
return headers != null ? headers : super.getHeaders();
}
private void configureRequest() {
}
}
Это может показаться излишним, но иметь все эти конструкторы довольно здорово, потому что у вас есть все случаи:
(Основной конструктор не предназначался для прямого использования, хотя, конечно, это возможно).
- Запрос с ответом, разобранным на POJO / Заголовки вручную установить / POJO для отправки
- Запрос с ответом анализируется на POJO / POJO для отправки
- Запрос с ответом анализируется на POJO / String для отправки
- Запрос с ответом, проанализированным на POJO (GET)
- Запрос с ответом, проанализированным на POJO (GET) / заголовки, установленные вручную
- Запрос без ответа (200 - пустое тело) / заголовки заданы вручную / POJO для отправки
- Запрос без ответа (200 - пустое тело) / POJO для отправки
- Запрос без ответа (200 - пустое тело) / строка для отправки
Конечно, для того, чтобы это работало, у вас должна быть Google GSON Lib; просто добавь:
compile 'com.google.code.gson:gson:x.y.z'
к вашим зависимостям (текущая версия 2.3.1
).
HashMap
вашем примере это как бы избыточно. Вы можете поместить «токен» прямо в aJSONObject
без промежуточной карты.