Android 初探Retrofit2.0.1(最新版)
Posted 一口仨馍
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android 初探Retrofit2.0.1(最新版)相关的知识,希望对你有一定的参考价值。
概述
看见这段开头,你应该听过Retrofit的大名但是没有真正的运用过,或者用过Retrofit1.9但是没有跟上更新的进度。无论什么原因,你都该好好了解下这篇博客。1.9和2.0.1版本差别很大。网上相关的教程很少,又因为是初探,所以本篇博客会用一个Demo说明Retrofit2.0.1的用法,
Retrofit GitHub地址:https://github.com/square/retrofit
Retrofit 项目主页:http://square.github.io/retrofit/
用一个Demo初探Retrofit
个人强烈建议先去浏览一遍上面的Retrofit 项目主页,会有助你了解Retrofit的基本用法和特性。
Demo结构概览
- 采用API Store里的天气接口,完整接口(带参数)地址:http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45
- MainActivity,没做封装,请求相关的逻辑调用都写在这里
- bean目录,GSON解析数据所用到的类
- service目录,定义请求相关参数及返回数据格式
使用Retrofit获取String类型的数据
添加依赖 compile ‘com.squareup.retrofit2:retrofit:2.0.1’
从service目录开始
首先需要定义一个类名为WeatherInfoService的interface
public interface WeatherInfoService
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call<ResponseBody> getString();
这里定义了一个名为getString()的抽象方法,由于这里没有指定返回类型,所以Call里面的泛型指定为ResponseBody。因为现在获取的是String类型的数据,所以暂时用不上Bean目录,直接进入MainActivity
MainActivity
private static final String BASE_URL = "http://v.juhe.cn/";
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn = (Button) findViewById(R.id.btn);
tv = (TextView) findViewById(R.id.tv);
btn.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
Retrofit retrofit = new Retrofit.Builder().baseUrl(BASE_URL).build();
WeatherInfoService weatherInfoService = retrofit.create(WeatherInfoService.class);
Call<ResponseBody> call = weatherInfoService.getString();
call.enqueue(new Callback<ResponseBody>()
@Override
public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response)
try
tv.setText(response.body().string());
catch (IOException e)
e.printStackTrace();
@Override
public void onFailure(Call<ResponseBody> call, Throwable t)
tv.setText(t.getMessage());
);
);
乍一看有点蒙,没关系我们慢慢来。
1. 首先我们创建了一个retrofit对象,这里需要注意下BASE_URL这个参数,如果@GET方法里的URl是完整的,则不拼接参数,否则拼接URL。
2. 接着我们将WeatherInfoService当做参数传递给retrofit的create()方法,返回值是WeatherInfoService的一个对象。这部分几乎是固定的,如果想简单封装可以在这部分做点手脚。
3. 然后调用WeatherInfoService对象的getString()方法,得到一个call
4. call有两种执行方式,一种是像上文中调用call.enqueue异步执行。既然有异步,那肯定得有同步。同步方法:Response<WeatherInfo> res = call.execute(); WeatherInfo info = res.body();
在主线程中调用同步方法可能会造成ANR。有兴趣自己测试下,不在本文之列。
至此我们已经有了一个使用Retrofit获取String数据的方法,但是通常情况下我们的服务器会返回JSON/XML格式的数据,以JSON居多。别急,Retrofit已经为我们考虑好了这一点。
使用Retrofit获取JSON类型的数据
添加依赖 compile ‘com.squareup.retrofit2:converter-gson:2.0.0-beta4’
这里使用GSON解析数据,当然你也可以换做其他的。支持列表如下
Gson: com.squareup.retrofit2:converter-gson Jackson:
com.squareup.retrofit2:converter-jackson Moshi:
com.squareup.retrofit2:converter-moshi Protobuf:
com.squareup.retrofit2:converter-protobuf Wire:
com.squareup.retrofit2:converter-wire Simple XML:
com.squareup.retrofit2:converter-simplexml
还是那个service
public interface WeatherInfoService
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call<ResponseBody> getString();
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call<WeatherInfo> getWeatherInfo();
同样的WeatherInfoService,方法名改为getWeatherInfo,指定返回Call中的泛型为WeatherInfo,下面简单展示下WeatherInfo这个Bean
等了好久的JavaBean
由于Bean有点多,而且都是GSON解析时对应的数据,都是getter/setter故这里只展示两个作为参考。
public class WeatherInfo
private String resultcode;
private String reason;
private Result result;
private String error_code;
...
public class Result
private Sk sk;
private Today today;
private List<Future> future;
...
对比接口 返回的数据,相信应该很容易理解,不再赘述。
还是那个MainActivity
btn2.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
final Gson gson = new GsonBuilder().create();
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.addConverterFactory(GsonConverterFactory.create(gson))
.build();
WeatherInfoService weatherInfoService = retrofit.create(WeatherInfoService.class);
Call<WeatherInfo> call= weatherInfoService.getWeatherInfo();
call.enqueue(new Callback<WeatherInfo>()
@Override
public void onResponse(Call<WeatherInfo> call, Response<WeatherInfo> response)
WeatherInfo info = response.body();
Log.i(TAG,gson.toJson(info));
tv.setText(gson.toJson(info));
@Override
public void onFailure(Call<WeatherInfo> call, Throwable t)
tv.setText("error"+t.getMessage());
);
);
细心的同学会注意到创建Retrofit对象的时候多了一行代码.addConverterFactory(GsonConverterFactory.create(gson))
和想象中一样添加了GSON解析器。不过是两行代码的事情,居然就可以直接返回GSON解析后的对象了。Retrofit确实方便,而且可拓展性杠杠的。
请求带参数
只需修改WeatherInfoService中的方法,完整的WeatherInfoService代码如下
package com.dyk.retrofit2.service;
import com.dyk.retrofit2.bean.WeatherInfo;
import okhttp3.ResponseBody;
import retrofit2.Call;
import retrofit2.http.GET;
import retrofit2.http.Query;
/**
* Created by dyk on 2016/4/9.
*/
public interface WeatherInfoService
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call<ResponseBody> getString();
@GET("/weather/index?/")
Call<WeatherInfo> getWeatherInfo(@Query("format") String format, @Query("cityname") String cityname, @Query("key") String key);
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call<WeatherInfo> getWeatherInfo();
MainActivity中调用的时候自然要传入相应的参数,MainActivity全部代码如下
package com.dyk.retrofit2;
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import com.dyk.retrofit2.bean.WeatherInfo;
import com.dyk.retrofit2.service.WeatherInfoService;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import java.io.IOException;
import okhttp3.ResponseBody;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class MainActivity extends Activity
private static final String TAG = "retrofit2";
private static final String BASE_URL = "http://v.juhe.cn/";
private static final String FORMAT = "2";
private static final String CITYNAME = "北京";
private static final String KEY = "b952ad7acbc7415f3f3c9bf274e39c45";
private Button btn;
private Button btn2;
private TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn = (Button) findViewById(R.id.btn);
tv = (TextView) findViewById(R.id.tv);
btn.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
Retrofit retrofit = new Retrofit.Builder().baseUrl(BASE_URL).build();
WeatherInfoService weatherInfoService = retrofit.create(WeatherInfoService.class);
Call<ResponseBody> call = weatherInfoService.getString();
call.enqueue(new Callback<ResponseBody>()
@Override
public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response)
try
tv.setText(response.body().string());
catch (IOException e)
e.printStackTrace();
@Override
public void onFailure(Call<ResponseBody> call, Throwable t)
tv.setText(t.getMessage());
);
);
btn2 = (Button) findViewById(R.id.btn2);
btn2.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
final Gson gson = new GsonBuilder().create();
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.addConverterFactory(GsonConverterFactory.create(gson))
.build();
WeatherInfoService weatherInfoService = retrofit.create(WeatherInfoService.class);
Call<WeatherInfo> call= weatherInfoService.getWeatherInfo(FORMAT, CITYNAME, KEY);
call.enqueue(new Callback<WeatherInfo>()
@Override
public void onResponse(Call<WeatherInfo> call, Response<WeatherInfo> response)
WeatherInfo info = response.body();
Log.i(TAG,gson.toJson(info));
tv.setText(gson.toJson(info));
@Override
public void onFailure(Call<WeatherInfo> call, Throwable t)
tv.setText("error"+t.getMessage());
);
);
结束语
很早就有闻Retrofit大名,终究没能一见。周末没事找Retrofit练练手,顺便学点新东西,结果网上极少有2.0.1版本(目前最新版)的教程。写完Demo后分享给大家,由于第一次接触Retrofit2.0.1,深度可能不是很够,但一定是最新的。后续还会继续深入了解Retrofit,希望大家能够关注。
以上是关于Android 初探Retrofit2.0.1(最新版)的主要内容,如果未能解决你的问题,请参考以下文章