看见这段开头,你应该听过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/
个人强烈建议先去浏览一遍上面的Retrofit 项目主页,会有助你了解Retrofit的基本用法和特性。
添加依赖 compile ‘com.squareup.retrofit2:retrofit:2.0.1’
首先需要定义一个类名为WeatherInfoService的interface
public interface WeatherInfoService {
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call getString();
}
这里定义了一个名为getString()的抽象方法,由于这里没有指定返回类型,所以Call里面的泛型指定为ResponseBody。因为现在获取的是String类型的数据,所以暂时用不上Bean目录,直接进图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 call = weatherInfoService.getString();
call.enqueue(new Callback() {
@Override
public void onResponse(Call call, Response response) {
try {
tv.setText(response.body().string());
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public void onFailure(Call 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
4.0后不能在主线程访问网络,所以同步需要新开线程。有兴趣自己测试下,不在本文之列。
至此我们已经有了一个使用Retrofit获取String数据的方法,但是通常情况下我们的服务器会返回JSON/XML格式的数据,以JSON居多。别急,Retrofit已经为我们考虑好了这一点。
添加依赖 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
public interface WeatherInfoService {
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call getString();
@GET("http://v.juhe.cn/weather/index?format=2&cityname=北京&key=b952ad7acbc7415f3f3c9bf274e39c45")
Call getWeatherInfo();
}
同样的WeatherInfoService,方法名改为getWeatherInfo,指定返回Call中的泛型为WeatherInfo,下面简单展示下WeatherInfo这个Bean
由于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;
...
}
对比接口 返回的数据,相信应该很容易理解,不再赘述。
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 call= weatherInfoService.getWeatherInfo();
call.enqueue(new Callback() {
@Override
public void onResponse(Call call, Response response) {
WeatherInfo info = response.body();
Log.i(TAG,gson.toJson(info));
tv.setText(gson.toJson(info));
}
@Override
public void onFailure(Call 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 getString();
@GET("/weather/index?/")
Call 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 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 call = weatherInfoService.getString();
call.enqueue(new Callback() {
@Override
public void onResponse(Call call, Response response) {
try {
tv.setText(response.body().string());
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public void onFailure(Call 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 call= weatherInfoService.getWeatherInfo(FORMAT, CITYNAME, KEY);
call.enqueue(new Callback() {
@Override
public void onResponse(Call call, Response response) {
WeatherInfo info = response.body();
Log.i(TAG,gson.toJson(info));
tv.setText(gson.toJson(info));
}
@Override
public void onFailure(Call call, Throwable t) {
tv.setText("error"+t.getMessage());
}
});
}
});
}
}
很早就有闻Retrofit大名,终究没能一见。周末没事找Retrofit练练手,顺便学点新东西,结果网上极少有2.0.1版本(目前最新版)的教程。写完Demo后分享给大家,由于第一次接触Retrofit2.0.1,深度可能不是很够,但一定是最新的。后续还会继续深入了解Retrofit,希望大家能够关注。