Retrofit Android Example Tutorial
Welcome to Retrofit Android Example Tutorial. Today we’ll use the Retrofit library developed by Square to handle REST API calls in our android application.
Retrofit Android
Retrofit is type-safe REST client for Android and Java which aims to make it easier to consume RESTful web services. We’ll not go into the details of Retrofit 1.x versions and jump onto Retrofit 2 directly which has a lot of new features and a changed internal API compared to the previous versions.
Retrofit 2 by default leverages OkHttp as the networking layer and is built on top of it.
Retrofit automatically serialises the JSON response using a POJO(Plain Old Java Object) which must be defined in advanced for the JSON Structure. To serialise JSON we need a converter to convert it into Gson first. We need to add the following dependencies in our
build.grade
file.compile 'com.squareup.retrofit2:retrofit:2.1.0' compile 'com.google.code.gson:gson:2.6.2' compile 'com.squareup.retrofit2:converter-gson:2.1.0'
OkHttp dependency is already shipped with Retrofit 2 dependency. If you wish to use a separate OkHttp dependency, you should exclude the OkHttp dependency from Retrofit 2 as:
compile ('com.squareup.retrofit2:retrofit:2.1.0') { // exclude Retrofit’s OkHttp dependency module and define your own module import exclude module: 'okhttp' } compile 'com.google.code.gson:gson:2.6.2' compile 'com.squareup.retrofit2:converter-gson:2.1.0' compile 'com.squareup.okhttp3:logging-interceptor:3.4.1' compile 'com.squareup.okhttp3:okhttp:3.4.1'
- The logging-interceptor generates a log string of the entire response that’s returned.
- There are other converters to parse the JSON to the necessary type. A few of them are listed below.
- Jackson :
com.squareup.retrofit2:converter-jackson:2.1.0
- Moshi :
com.squareup.retrofit2:converter-moshi:2.1.0
- Protobuf :
com.squareup.retrofit2:converter-protobuf:2.1.0
- Wire :
com.squareup.retrofit2:converter-wire:2.1.0
- Simple XML :
com.squareup.retrofit2:converter-simplexml:2.1.0
Add the permission to access internet in the AndroidManifest.xml file.
OkHttp Interceptors
Interceptors are a powerful mechanism present in OkHttp that can monitor, rewrite, and retry calls.
Interceptors can be majorly divided into two categories:
Interceptors can be majorly divided into two categories:
- Application Interceptors : To register an application interceptor, we need to call
addInterceptor()
onOkHttpClient.Builder
- Network Interceptors : To register a Network Interceptor, invoke
addNetworkInterceptor()
instead ofaddInterceptor()
Setting Up the Retrofit Interface
package com.journaldev.retrofitintro; import com.journaldev.retrofitintro.pojo.MultipleResource; import okhttp3.OkHttpClient; import okhttp3.logging.HttpLoggingInterceptor; import retrofit2.Retrofit; import retrofit2.converter.gson.GsonConverterFactory; class APIClient { private static Retrofit retrofit = null; static Retrofit getClient() { HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor(); interceptor.setLevel(HttpLoggingInterceptor.Level.BODY); OkHttpClient client = new OkHttpClient.Builder().addInterceptor(interceptor).build(); retrofit = new Retrofit.Builder() .baseUrl("https://reqres.in") .addConverterFactory(GsonConverterFactory.create()) .client(client) .build(); return retrofit; } }
The
getClient()
method in the above code will be called every time while setting up a Retrofit interface. Retrofit provides with a list of annotations for each of the HTTP methods: @GET, @POST, @PUT, @DELETE, @PATCH or @HEAD
.
Let’s see how our
APIInterface.java
class looks like.package com.journaldev.retrofitintro; import com.journaldev.retrofitintro.pojo.MultipleResource; import com.journaldev.retrofitintro.pojo.User; import com.journaldev.retrofitintro.pojo.UserList; import retrofit2.Call; import retrofit2.http.Body; import retrofit2.http.Field; import retrofit2.http.FormUrlEncoded; import retrofit2.http.GET; import retrofit2.http.POST; import retrofit2.http.Query; interface APIInterface { @GET("/api/unknown") Call<MultipleResource> doGetListResources(); @POST("/api/users") Call<User> createUser(@Body User user); @GET("/api/users?") Call<UserList> doGetUserList(@Query("page") String page); @FormUrlEncoded @POST("/api/users?") Call<UserList> doCreateUserWithField(@Field("name") String name, @Field("job") String job); }
In the above class, we’ve defined some methods that perform HTTP requests with annotation.
We’ve used a few test APIs from here
We’ve used a few test APIs from here
@GET("/api/unknown")
calls doGetListResources();
.doGetListResources()
is the method name. MultipleResource.java
is a Model POJO class for our response object that’s used to map the response parameters to their respective variables. These POJO class act as the method return type.
A simple POJO class for
MultipleResources.java
is given below.package com.journaldev.retrofitintro.pojo; import com.google.gson.annotations.SerializedName; import java.util.ArrayList; import java.util.List; public class MultipleResource { @SerializedName("page") public Integer page; @SerializedName("per_page") public Integer perPage; @SerializedName("total") public Integer total; @SerializedName("total_pages") public Integer totalPages; @SerializedName("data") public List data = null; public class Datum { @SerializedName("id") public Integer id; @SerializedName("name") public String name; @SerializedName("year") public Integer year; @SerializedName("pantone_value") public String pantoneValue; } }
@SerializedName
annotation is used to specify the name of the field that’s in the JSON Response.
To create a POJO class for each response, we can go to http://www.jsonschema2pojo.org/and paste the json response structure as shown in the image below.
Preview the POJO class and copy it into your Android Studio Project Structure.
The POJO classes are wrapped into a typed Retrofit
Call
class.
Note: A JSONArray is serialised a List of Objects in the POJO classes
Method Parameters : There are a wide variety of possible options of parameters to pass inside a method:
@Body
– Sends Java objects as request body.@Url
– use dynamic URLs.@Query
– We can simply add a method parameter with @Query() and a query parameter name, describing the type. To URL encode a query use the form:@Query(value = "auth_token",encoded = true) String auth_token
@Field
– send data as form-urlencoded. This requires a@FormUrlEncoded
annotation attached with the method.
The@Field
parameter works only with a POST
Note: @Field requires a mandatory parameter. In cases when @Field is optional, we can use @Query instead and pass a null value.
Retrofit Android Example Project Structure
The pojo package defines four model classes for each of the API endpoint responses defined in the APIInterface.java class.
User.java
package com.journaldev.retrofitintro.pojo; import com.google.gson.annotations.SerializedName; public class User { @SerializedName("name") public String name; @SerializedName("job") public String job; @SerializedName("id") public String id; @SerializedName("createdAt") public String createdAt; public User(String name, String job) { this.name = name; this.job = job; } }
The above class is used to create the Response Body for the
createUser()
methodUserList.java
package com.journaldev.retrofitintro.pojo; import com.google.gson.annotations.SerializedName; import java.util.ArrayList; import java.util.List; public class UserList { @SerializedName("page") public Integer page; @SerializedName("per_page") public Integer perPage; @SerializedName("total") public Integer total; @SerializedName("total_pages") public Integer totalPages; @SerializedName("data") public List data = new ArrayList(); public class Datum { @SerializedName("id") public Integer id; @SerializedName("first_name") public String first_name; @SerializedName("last_name") public String last_name; @SerializedName("avatar") public String avatar; } }
CreateUserResponse.java
package com.journaldev.retrofitintro.pojo; import com.google.gson.annotations.SerializedName; public class CreateUserResponse { @SerializedName("name") public String name; @SerializedName("job") public String job; @SerializedName("id") public String id; @SerializedName("createdAt") public String createdAt; }
The
MainActivity.java
is where we call each of the API endpoints defined in the Interface class and display each of the fields in a Toast/TextView.package com.journaldev.retrofitintro; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.widget.TextView; import android.widget.Toast; import com.journaldev.retrofitintro.pojo.CreateUserResponse; import com.journaldev.retrofitintro.pojo.MultipleResource; import com.journaldev.retrofitintro.pojo.User; import com.journaldev.retrofitintro.pojo.UserList; import java.util.List; import retrofit2.Call; import retrofit2.Callback; import retrofit2.Response; public class MainActivity extends AppCompatActivity { TextView responseText; APIInterface apiInterface; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); responseText = (TextView) findViewById(R.id.responseText); apiInterface = APIClient.getClient().create(APIInterface.class); /** GET List Resources **/ Call call = apiInterface.doGetListResources(); call.enqueue(new Callback() { @Override public void onResponse(Call call, Response response) { Log.d("TAG",response.code()+""); String displayResponse = ""; MultipleResource resource = response.body(); Integer text = resource.page; Integer total = resource.total; Integer totalPages = resource.totalPages; List datumList = resource.data; displayResponse += text + " Page\n" + total + " Total\n" + totalPages + " Total Pages\n"; for (MultipleResource.Datum datum : datumList) { displayResponse += datum.id + " " + datum.name + " " + datum.pantoneValue + " " + datum.year + "\n"; } responseText.setText(displayResponse); } @Override public void onFailure(Call call, Throwable t) { call.cancel(); } }); /** Create new user **/ User user = new User("morpheus", "leader"); Call call1 = apiInterface.createUser(user); call1.enqueue(new Callback() { @Override public void onResponse(Call call, Response response) { User user1 = response.body(); Toast.makeText(getApplicationContext(), user1.name + " " + user1.job + " " + user1.id + " " + user1.createdAt, Toast.LENGTH_SHORT).show(); } @Override public void onFailure(Call call, Throwable t) { call.cancel(); } }); /** GET List Users **/ Call call2 = apiInterface.doGetUserList("2"); call2.enqueue(new Callback() { @Override public void onResponse(Call call, Response response) { UserList userList = response.body(); Integer text = userList.page; Integer total = userList.total; Integer totalPages = userList.totalPages; List datumList = userList.data; Toast.makeText(getApplicationContext(), text + " page\n" + total + " total\n" + totalPages + " totalPages\n", Toast.LENGTH_SHORT).show(); for (UserList.Datum datum : datumList) { Toast.makeText(getApplicationContext(), "id : " + datum.id + " name: " + datum.first_name + " " + datum.last_name + " avatar: " + datum.avatar, Toast.LENGTH_SHORT).show(); } } @Override public void onFailure(Call call, Throwable t) { call.cancel(); } }); /** POST name and job Url encoded. **/ Call call3 = apiInterface.doCreateUserWithField("morpheus","leader"); call3.enqueue(new Callback() { @Override public void onResponse(Call call, Response response) { UserList userList = response.body(); Integer text = userList.page; Integer total = userList.total; Integer totalPages = userList.totalPages; List datumList = userList.data; Toast.makeText(getApplicationContext(), text + " page\n" + total + " total\n" + totalPages + " totalPages\n", Toast.LENGTH_SHORT).show(); for (UserList.Datum datum : datumList) { Toast.makeText(getApplicationContext(), "id : " + datum.id + " name: " + datum.first_name + " " + datum.last_name + " avatar: " + datum.avatar, Toast.LENGTH_SHORT).show(); } } @Override public void onFailure(Call call, Throwable t) { call.cancel(); } }); } }
apiInterface = APIClient.getClient().create(APIInterface.class);
is used to instantiate the APIClient.To map the Model class to the response we use:
MultipleResource resource = response.body();
Running the application would call each of the endpoints and display a Toast message for them accordingly.
This brings an end to Retrofit android example tutorial. You can download the Android Retrofit example project from the link below.
https://api.foursquare.com/v2/venues/explore?client_id=TDQG4LF1M1O5SZESHJ42BRSZQDPJY4B53NQN5RDTRRZZPNYM&client_secret=PGLQ4RV5R0FDIPHAPJO4RT3KJXO02ARZ4LQ3NWS3TPFIUUJ0&v=20130815&ne=19.241143,72.994881&query=history&sw=18.912417,72.82397
I need to fetch venue name, lat lng ,checkin counts, rating .How do I go about it?
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
@SerializedName(“marked”)
@Expose
public Boolean marked;
@SerializedName(“lastCheckinExpiredAt”)
@Expose
public Integer lastCheckinExpiredAt;
———————————–com.example.Category.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String id;
@SerializedName(“name”)
@Expose
public String name;
@SerializedName(“pluralName”)
@Expose
public String pluralName;
@SerializedName(“shortName”)
@Expose
public String shortName;
@SerializedName(“icon”)
@Expose
public Icon icon;
@SerializedName(“primary”)
@Expose
public Boolean primary;
———————————–com.example.Contact.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String phone;
@SerializedName(“formattedPhone”)
@Expose
public String formattedPhone;
@SerializedName(“twitter”)
@Expose
public String twitter;
———————————–com.example.Entity.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public List indices = null;
@SerializedName(“type”)
@Expose
public String type;
———————————–com.example.Example.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Meta meta;
@SerializedName(“response”)
@Expose
public Response response;
———————————–com.example.Filter.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String name;
@SerializedName(“key”)
@Expose
public String key;
———————————–com.example.Group.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public String type;
@SerializedName(“name”)
@Expose
public String name;
@SerializedName(“items”)
@Expose
public List items = null;
———————————–com.example.HereNow.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
@SerializedName(“summary”)
@Expose
public String summary;
@SerializedName(“groups”)
@Expose
public List groups = null;
———————————–com.example.Hours.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Boolean isOpen;
@SerializedName(“isLocalHoliday”)
@Expose
public Boolean isLocalHoliday;
@SerializedName(“status”)
@Expose
public String status;
———————————–com.example.Icon.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String prefix;
@SerializedName(“suffix”)
@Expose
public String suffix;
———————————–com.example.Item.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public Reasons reasons;
@SerializedName(“venue”)
@Expose
public Venue venue;
@SerializedName(“tips”)
@Expose
public List tips = null;
@SerializedName(“referralId”)
@Expose
public String referralId;
———————————–com.example.Item_.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String summary;
@SerializedName(“type”)
@Expose
public String type;
@SerializedName(“reasonName”)
@Expose
public String reasonName;
———————————–com.example.LabeledLatLng.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String label;
@SerializedName(“lat”)
@Expose
public Float lat;
@SerializedName(“lng”)
@Expose
public Float lng;
———————————–com.example.Likes.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
@SerializedName(“groups”)
@Expose
public List groups = null;
@SerializedName(“summary”)
@Expose
public String summary;
———————————–com.example.Location.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public String address;
@SerializedName(“lat”)
@Expose
public Float lat;
@SerializedName(“lng”)
@Expose
public Float lng;
@SerializedName(“labeledLatLngs”)
@Expose
public List labeledLatLngs = null;
@SerializedName(“postalCode”)
@Expose
public String postalCode;
@SerializedName(“cc”)
@Expose
public String cc;
@SerializedName(“city”)
@Expose
public String city;
@SerializedName(“state”)
@Expose
public String state;
@SerializedName(“country”)
@Expose
public String country;
@SerializedName(“formattedAddress”)
@Expose
public List formattedAddress = null;
@SerializedName(“crossStreet”)
@Expose
public String crossStreet;
———————————–com.example.Meta.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Integer code;
@SerializedName(“requestId”)
@Expose
public String requestId;
———————————–com.example.Ne.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Float lat;
@SerializedName(“lng”)
@Expose
public Float lng;
———————————–com.example.Photo.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String prefix;
@SerializedName(“suffix”)
@Expose
public String suffix;
———————————–com.example.Photo_.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String id;
@SerializedName(“createdAt”)
@Expose
public Integer createdAt;
@SerializedName(“source”)
@Expose
public Source source;
@SerializedName(“prefix”)
@Expose
public String prefix;
@SerializedName(“suffix”)
@Expose
public String suffix;
@SerializedName(“width”)
@Expose
public Integer width;
@SerializedName(“height”)
@Expose
public Integer height;
@SerializedName(“visibility”)
@Expose
public String visibility;
———————————–com.example.Photos.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
@SerializedName(“groups”)
@Expose
public List groups = null;
———————————–com.example.Price.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Integer tier;
@SerializedName(“message”)
@Expose
public String message;
@SerializedName(“currency”)
@Expose
public String currency;
———————————–com.example.Reasons.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
@SerializedName(“items”)
@Expose
public List items = null;
———————————–com.example.Response.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public SuggestedFilters suggestedFilters;
@SerializedName(“headerLocation”)
@Expose
public String headerLocation;
@SerializedName(“headerFullLocation”)
@Expose
public String headerFullLocation;
@SerializedName(“headerLocationGranularity”)
@Expose
public String headerLocationGranularity;
@SerializedName(“query”)
@Expose
public String query;
@SerializedName(“totalResults”)
@Expose
public Integer totalResults;
@SerializedName(“suggestedBounds”)
@Expose
public SuggestedBounds suggestedBounds;
@SerializedName(“groups”)
@Expose
public List groups = null;
———————————–com.example.Source.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String name;
@SerializedName(“url”)
@Expose
public String url;
———————————–com.example.Stats.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Integer checkinsCount;
@SerializedName(“usersCount”)
@Expose
public Integer usersCount;
@SerializedName(“tipCount”)
@Expose
public Integer tipCount;
———————————–com.example.SuggestedBounds.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Ne ne;
@SerializedName(“sw”)
@Expose
public Sw sw;
———————————–com.example.SuggestedFilters.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public String header;
@SerializedName(“filters”)
@Expose
public List filters = null;
———————————–com.example.Sw.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Float lat;
@SerializedName(“lng”)
@Expose
public Float lng;
———————————–com.example.Tip.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public String id;
@SerializedName(“createdAt”)
@Expose
public Integer createdAt;
@SerializedName(“text”)
@Expose
public String text;
@SerializedName(“entities”)
@Expose
public List entities = null;
@SerializedName(“type”)
@Expose
public String type;
@SerializedName(“canonicalUrl”)
@Expose
public String canonicalUrl;
@SerializedName(“logView”)
@Expose
public Boolean logView;
@SerializedName(“agreeCount”)
@Expose
public Integer agreeCount;
@SerializedName(“disagreeCount”)
@Expose
public Integer disagreeCount;
@SerializedName(“todo”)
@Expose
public Todo todo;
@SerializedName(“user”)
@Expose
public User user;
@SerializedName(“likes”)
@Expose
public Likes likes;
@SerializedName(“photo”)
@Expose
public Photo_ photo;
@SerializedName(“photourl”)
@Expose
public String photourl;
———————————–com.example.Todo.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public Integer count;
———————————–com.example.User.java———————————–
import com.google.gson.annotations.SerializedName;
@Expose
public String id;
@SerializedName(“firstName”)
@Expose
public String firstName;
@SerializedName(“lastName”)
@Expose
public String lastName;
@SerializedName(“gender”)
@Expose
public String gender;
@SerializedName(“photo”)
@Expose
public Photo photo;
———————————–com.example.Venue.java———————————–
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
@Expose
public String id;
@SerializedName(“name”)
@Expose
public String name;
@SerializedName(“contact”)
@Expose
public Contact contact;
@SerializedName(“location”)
@Expose
public Location location;
@SerializedName(“categories”)
@Expose
public List categories = null;
@SerializedName(“verified”)
@Expose
public Boolean verified;
@SerializedName(“stats”)
@Expose
public Stats stats;
@SerializedName(“rating”)
@Expose
public Float rating;
@SerializedName(“ratingSignals”)
@Expose
public Integer ratingSignals;
@SerializedName(“allowMenuUrlEdit”)
@Expose
public Boolean allowMenuUrlEdit;
@SerializedName(“beenHere”)
@Expose
public BeenHere beenHere;
@SerializedName(“hours”)
@Expose
public Hours hours;
@SerializedName(“photos”)
@Expose
public Photos photos;
@SerializedName(“hereNow”)
@Expose
public HereNow hereNow;
@SerializedName(“venueRatingBlacklisted”)
@Expose
public Boolean venueRatingBlacklisted;
@SerializedName(“url”)
@Expose
public String url;
@SerializedName(“price”)
@Expose
public Price price;