Wiem, że ten wątek jest dość stary, ale miałem ten problem i wymyśliłem fajne rozwiązanie, które może być bardzo przydatne dla wielu, ponieważ poprawia / rozszerza bibliotekę Volley pod wieloma względami.
Zauważyłem kilka nieobsługiwanych funkcji Volley po wyjęciu z pudełka:
- To
JSONObjectRequest
nie jest idealne: JSON
na końcu musisz się spodziewać (patrz Response.Listener<JSONObject>
).
- A co z pustymi odpowiedziami (tylko ze statusem 200)?
- Co mam zrobić, jeśli chcę otrzymać POJO bezpośrednio z
ResponseListener
?
Z grubsza skompilowałem wiele rozwiązań w dużej klasie ogólnej, aby uzyskać rozwiązanie całego zacytowanego problemu.
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() {
}
}
Może się to wydawać trochę przesadzone, ale fajnie jest mieć wszystkich tych konstruktorów, ponieważ masz wszystkie przypadki:
(Główny konstruktor nie miał być używany bezpośrednio, chociaż jest to oczywiście możliwe).
- Żądanie z odpowiedzią przeanalizowane do POJO / nagłówki ręcznie ustawione / POJO do wysłania
- Żądanie z odpowiedzią przetworzone do POJO / POJO w celu wysłania
- Żądanie z odpowiedzią przeanalizowane do POJO / String do wysłania
- Żądanie z odpowiedzią przeanalizowane do POJO (GET)
- Żądanie z odpowiedzią przeanalizowane do POJO (GET) / nagłówki ustawione ręcznie
- Żądanie bez odpowiedzi (200 - pusta treść) / nagłówki ustawione ręcznie / POJO do wysłania
- Żądanie bez odpowiedzi (200 - pusta treść) / POJO do wysłania
- Żądanie bez odpowiedzi (200 - pusta treść) / ciąg do wysłania
Oczywiście, aby to działało, musisz mieć GSON Lib Google; poprostu dodaj:
compile 'com.google.code.gson:gson:x.y.z'
do twoich zależności (aktualna wersja to 2.3.1
).
HashMap
twoim przykładzie jest to trochę zbyteczne. Możesz umieścić „token” bezpośrednio wJSONObject
mapie bez pośredniej mapy.