项目作者: hayden621

项目描述 :
A tutorial on how to use volley + gson with android
高级语言: Java
项目地址: git://github.com/hayden621/volley.git
创建时间: 2018-04-22T05:52:56Z
项目社区:https://github.com/hayden621/volley

开源协议:GNU General Public License v3.0

下载


A tutorial on how to use volley and gson with android. Thanks to https://reqres.in/api/ (https://github.com/benhowdle89/reqres) for API demostration

Pre-request

Create empty android project

Preparation

  1. Add volley library to gradle

    1. implementation 'com.android.volley:volley:1.0.0'
  2. Add gson library to gradle

    1. implementation 'com.google.code.gson:gson:2.4'
  3. Add internet permission to manifest.xml

Core Class

  1. ApiPool - Centralized pool for api request queue
  2. CookieHelper - For handling cookies in request header (not normally required)
  3. GeneralRequest - Mother class for all request object
  4. GeneralResponse - Mother class for all response object
  5. GenericApiHelper - A generic helper class to make api call
  6. SharedPreferenceHelper - For storing cookies in shared preference

Sample API call with Core classes

At MainActivity, make API call to https://reqres.in/api/users/2 with response as below, then display returned first and last name

  1. //json
  2. {
  3. "data": {
  4. "id": 2,
  5. "first_name": "Janet",
  6. "last_name": "Weaver",
  7. "avatar": "https://s3.amazonaws.com/uifaces/faces/twitter/josephstein/128.jpg"
  8. }
  9. }
  1. Create SampleApiHelper which extends GenericApiHelper
  2. Overload call() method with pre-defined api domain address and api time out
  3. Create interface for SampleApiCallbackListener which extends ApiResponseListener
  4. Create Data and SampleResponse according to api response in step 1, you can use (http://www.jsonschema2pojo.org/) to generate pojo from json
  5. At MainActivity, implements SampleApiCallbackListener and generates the override methods accordingly
  6. Use SampleApiHelper to make call to https://reqres.in/api/users/2, which is a self defined response code
  7. The response code is used to identify which API it responses to(as multiple api call can be made)
  8. Handle the response at onApiResponse() and onApiError()

Understanding core classes

GenericApiHelper

  1. //JAVA method call()
  2. /**
  3. * method to make REST API request
  4. * @param context
  5. * @param method: Request.Method.*
  6. * @param domain: API domain address
  7. * @param endpoint: API endpoint
  8. * @param header: request header
  9. * @param requestParams: parameters append to url
  10. * @param requestObject: GeneralRequest object which will be parse to json by gson
  11. * @param responseObjectClazz: GeneralReponse class which allow the api response to be parse back to java object
  12. * @param responseCode: User definded integer to identify which API it is responding to
  13. * @param outTime: API connection time out
  14. * @param apiResponseListener: API callback listener for listening to success and fail api call
  15. */
  16. protected void call(final Context context, int method, final String domain, final String endpoint, final Map<String,String> header, final Map<String,String> requestParams, final GeneralRequest requestObject, final Class<?> responseObjectClazz, final int responseCode, final int outTime, final ApiResponseListener apiResponseListener) {
  17. String url = decorateUrl(context,domain, endpoint,requestParams);
  18. Log.d(TAG, "url: " + url);
  19. // Request a string response from the provided URL.
  20. StringRequest stringRequest = new StringRequest(method, url,
  21. new Response.Listener<String>() {
  22. @Override
  23. //On API success return
  24. public void onResponse(String response) {
  25. Log.d(TAG, endpoint + "response:" + response);
  26. if(apiResponseListener != null) {
  27. //try parsing json response to response object
  28. try {
  29. Object responseObj = gson.fromJson(response,(Class<?>) responseObjectClazz);
  30. Log.d(TAG, "responseObj:" + responseObj.toString());
  31. //cast response object to GeneralResponse and return to listener
  32. if(responseObj instanceof GeneralResponse) {
  33. GeneralResponse generalResponse = (GeneralResponse) responseObj;
  34. apiResponseListener.onApiResponse(responseCode, generalResponse);
  35. }
  36. else {
  37. String errorMsg = "error parsing response to GeneralResponse";
  38. apiResponseListener.onApiError(responseCode, errorMsg);
  39. }
  40. }
  41. catch(JsonSyntaxException e) {
  42. apiResponseListener.onApiError(responseCode, e.toString());
  43. }
  44. }
  45. }
  46. }, new Response.ErrorListener() {
  47. @Override
  48. public void onErrorResponse(VolleyError error) {
  49. Log.e(TAG, "error calling:[" + endpoint + "], error:" + error.toString());
  50. if(apiResponseListener != null) {
  51. apiResponseListener.onApiError(responseCode, error.toString());
  52. }
  53. }
  54. }) {
  55. @Override
  56. public byte[] getBody() throws AuthFailureError {
  57. if(requestObject != null) {
  58. try {
  59. //parse request object to json format and send as request body
  60. return gson.toJson(requestObject).getBytes();
  61. } catch (Exception e) {
  62. Log.e(TAG, "error parsing request body to json");
  63. }
  64. }
  65. return super.getBody();
  66. }
  67. @Override
  68. public Map<String, String> getHeaders() throws AuthFailureError {
  69. Map<String,String> params = new HashMap<>();
  70. params.put("Content-Type","application/json; charset=utf-8");
  71. if(header != null) {
  72. for(Map.Entry<String, String> entry : header.entrySet()) {
  73. params.put(entry.getKey(), entry.getValue());
  74. }
  75. }
  76. params = CookieHelper.getInstance().addSessionCookie(context, params);
  77. return params;
  78. }
  79. @Override
  80. public String getBodyContentType() {
  81. return "application/json; charset=utf-8";
  82. }
  83. @Override
  84. protected Response<String> parseNetworkResponse(NetworkResponse response) {
  85. // since we don't know which of the two underlying network vehicles
  86. // will Volley use, we have to handle and store session cookies manually
  87. CookieHelper.getInstance().checkSessionCookie(context, response.headers);
  88. return super.parseNetworkResponse(response);
  89. }
  90. };
  91. stringRequest.setRetryPolicy(new DefaultRetryPolicy(
  92. outTime,
  93. DefaultRetryPolicy.DEFAULT_MAX_RETRIES,
  94. DefaultRetryPolicy.DEFAULT_MAX_RETRIES
  95. ));
  96. ApiPool.getInstance(context).call(stringRequest);
  97. }

call() method is the core function to make api call. To use this method, you should create a child class of it and exposes the call method.

decorateUrl()

  1. private String decorateUrl(Context context, final String domain, final String endpoint, final Map<String,String> requestParams) {
  2. String url = domain + endpoint;
  3. if(requestParams != null) {
  4. url = url + "?";
  5. for(Map.Entry<String, String> entry : requestParams.entrySet()) {
  6. url = url + entry.getKey() + "=" + entry.getValue() + "&";
  7. }
  8. }
  9. return url;
  10. }

This will create the url path by concatenating the domain name endpoint and GET request params.
Request parameters should be in Map format and will be appended at the end of url in key=value format