您的位置:首页 > 移动开发 > Android开发

Android 初探Retrofit2.0.1(最新版)

2016-04-09 23:29 375 查看

概述

看见这段开头,你应该听过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,希望大家能够关注。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: