Retrofit 2.0 使用总结

来源:互联网 发布:创始于淘宝的女装品牌 编辑:程序博客网 时间:2024/05/29 10:33

这天,leader给了我一张图,然后让我调试后台接口,图片如下:
后台接口

我看这么简单,然后直接用浏览器测试了一下,然后返回图片如下:

这里写图片描述
然后花了一个下午时间测试,终于把接口调通了,但是居然解析不了其中的返回内容,,,,于是,在礼拜天我就花时间来研究了一下关于接口的开发!
本来我打算随便写一个 Java 类,然后用tomcat就可以测试后台了,但是我不知道如何实现上图的禁止get请求,于是问了一下这方面的大神:

我用spring mvc的时候,controler的requestmapping的method 的值设为post就会无视get请求

由于上面的知识点我都不会,于是我自动忽略了自己写后台的想法,接着打算在网上找一个可以测试后台的网站,结果找到一个 http://www.ouapi.com/ ,但是打开页面以后也不知道如何设置后台,后来询问大神得知——人家一般使用的是 postman 这个插件,于是我按照网上的介绍,赶紧安装了一个!测试了一下,还真的很好用,以图为证:
这里写图片描述
废话说了这么多,接下来开始干正事!介绍一下自己使用 Retrofit 2.0 的一些总结。第一步,肯定是得导包,这个随便百度一下 Retrofit 2.0 就可以翻出来了,

compile 'com.squareup.retrofit2:retrofit:2.0.2'  //retrofit:2.0compile 'com.squareup.retrofit2:converter-gson:2.0.2'   //retrofit:2.0封装下的gsoncompile 'com.squareup.okhttp3:logging-interceptor:3.2.0'    //retrofit:2.0下的okhttp3的请求日志

第二步就是实例化 retrofit ,这个基本上也是可以直接从网上复制下来就可以用

  HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();        interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);        OkHttpClient client = new OkHttpClient.Builder()                .addInterceptor(interceptor)                .retryOnConnectionFailure(true)                .connectTimeout(15, TimeUnit.SECONDS)                .addNetworkInterceptor(interceptor)                .build();        Gson gson = new GsonBuilder()                .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")                .create();        //获取实例        Retrofit retrofit = new Retrofit.Builder()                //设置OKHttpClient,如果不设置会提供一个默认的                .client(client)                //设置baseUrl,必须是以“/”结尾                .baseUrl(url)                //添加Gson转换器                .addConverterFactory(GsonConverterFactory.create(gson))                .build();

接下来就是关于各种网络请求的封装了,比如我们先请求一段本地的文件,测试一下 get 请求,简单来说我们的第三步需要如下步骤:

  1. 建立 interface ,并封装 get 请求
  2. 通过上面实例化的 Retrofit 对象,实例化上面的 interface 对象
  3. 调用上述封装的 get 请求

    通过上面的步骤,大概知道做什么以后,那么我们通过 postman 来测试一下这个接口:
    get 请求
    当然,大家也可以通过post来测试这个接口,待会我们就可以来测试,现在开始来执行上面的第一步——建立 interface ,并封装 get 请求:

public interface RetrofitInterface {    @GET("getdata.gson")    Call<Password> getPassWord();}

这里说一下 @GET(“getdata.gson”) 这里为什么会跟一个文件呢?因为上面说过,

设置baseUrl,必须是以“/”结尾(重要的事情只说一边!)

所以我们现在带着上面的提醒来执行第二步——通过上面实例化的 Retrofit 对象,实例化上面的 interface 对象:

/** * Retrofit实体类 * Created by asus on 2017/7/15. */public class RetrofitLoader {    private RetrofitInterface service;    public RetrofitLoader(String url){        HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();        interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);        OkHttpClient client = new OkHttpClient.Builder()                .addInterceptor(interceptor)                .retryOnConnectionFailure(true)                .connectTimeout(15, TimeUnit.SECONDS)                .addNetworkInterceptor(interceptor)                .build();        Gson gson = new GsonBuilder()                .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")                .create();        //获取实例        Retrofit retrofit = new Retrofit.Builder()                //设置OKHttpClient,如果不设置会提供一个默认的                .client(client)                //设置baseUrl,必须是以“/”结尾                .baseUrl(url)                //添加Gson转换器                .addConverterFactory(GsonConverterFactory.create(gson))                .build();        service = retrofit.create(RetrofitInterface.class);    }   public void getPassWord(Callback<Password> callback){        Call<Password> call = service.getPassWord();        //异步请求        call.enqueue(callback);        //同步请求        //1call.execute();    }}

上面的 RetrofitLoader 应该做一个单例,这里只是测试,下来后慢慢完善!接下来就是第三步——调用上述封装的 get 请求:

public class MainActivity extends AppCompatActivity {    private static final String TAG = "MainActivity";    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        final TextView textView = (TextView) findViewById(R.id.textview);        Thread thread = Thread.currentThread();        Log.e(TAG, "onCreate: "+ thread.getId());        new Thread(){            @Override            public void run() {                super.run();                Thread thread = Thread.currentThread();                Log.e(TAG, "run1: "+ thread.getId());                 new RetrofitLoader("http://192.168.199.237:8080/examples/").getPassWord(new Callback<Password>() {                     @Override                     public void onResponse(Call<Password> call,final Response<Password> response) {                         textView.setText(response.body().getData());                         Thread thread = Thread.currentThread();                         Log.e(TAG, "onResponse: "+ thread.getId());                         textView.post(new Runnable() {                             @Override                             public void run() {                                 Thread thread = Thread.currentThread();                                 Log.e(TAG, "run2: "+ thread.getId());                             }                         });                     }                     @Override                     public void onFailure(Call<Password> call, Throwable t) {                         Log.e(TAG, "onFailure: ",t );                     }                 });            }        }.start();    }}

接下来我们看看日志:

07-16 21:52:35.837 9174-9174/? E/MainActivity: onCreate: 107-16 21:52:35.838 9174-9195/? E/MainActivity: run1: 124807-16 21:52:36.076 9174-9174/? E/MainActivity: onResponse: 107-16 21:52:36.076 9174-9174/? E/MainActivity: onResponse: 7B7F25D86267923209A11145A4EBB3407-16 21:52:36.086 9174-9174/? E/MainActivity: run2: 1

这里说明一个问题:在线程里面开子线程请求网络(线程 ID :1248 ),然后异步返回的线程不是在当前所在的子线程,而是在主线程,至于网络请求的同步,这个就不记录测试了,其实我最想测试的是 Post 请求,而且是带参数的,因为工作中需要用到,但是在测试过程中发生问题了!
在讲问题之前先看看API吧,如图:
这里写图片描述
接下来我们先封装一个 Post 请求,不过这里不需要单独新建 interface 对象了,

/** * Created by asus on 2017/7/15. */public interface RetrofitInterface {    @GET("getdata.gson")    Call<Password> getPassWord();    @POST("LookUp")    Call<IdCardResult> checkIdCard(@Body IdCardInfo cardInfo);}/** * Retrofit实体类 * Created by asus on 2017/7/15. */public class RetrofitLoader {    private RetrofitInterface service;    public RetrofitLoader(String url){        HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();        interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);        OkHttpClient client = new OkHttpClient.Builder()                .addInterceptor(interceptor)                .retryOnConnectionFailure(true)                .connectTimeout(15, TimeUnit.SECONDS)                .addNetworkInterceptor(interceptor)                .build();        Gson gson = new GsonBuilder()                .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")                .create();        //获取实例        Retrofit retrofit = new Retrofit.Builder()                //设置OKHttpClient,如果不设置会提供一个默认的                .client(client)                //设置baseUrl,必须是以“/”结尾                .baseUrl(url)                //添加Gson转换器                .addConverterFactory(GsonConverterFactory.create(gson))                .build();        service = retrofit.create(RetrofitInterface.class);    }    public void getPassWord(Callback<Password> callback){        Call<Password> call = service.getPassWord();        //异步请求        call.enqueue(callback);        //同步请求        //1call.execute();    }    public void checkIdCard(IdCardInfo cardInfo, Callback<IdCardResult> callback){        service.checkIdCard(cardInfo).enqueue(callback);    }}

接下来我们看看调用过程中是不是有传说的异常?
访问成功了
访问成功了?那么我们继续来看看访问不成功,只要将post方法修改一下,这个方法就运行不起来:

Call<Response> checkIdCard(@Body IdCardInfo cardInfo);

这里写图片描述

虽然 Body 标签完成了访问,但是返回码是 1001,返回码是 key 值错误,但是这个值我确定没有复制错,具体是怎么回事呢?看看日志:
这里写图片描述

对 http 来说参数之间的分割只有一种 & ,这里的,是错误的,所以服务端报错,那么这里该如何修改呢?

 @FormUrlEncoded    @POST("LookUp")    Call<IdCardResult> checkIdCard(@FieldMap Map<String, String> infos); public void checkIdCard(IdCardInfo cardInfo, Callback<IdCardResult> callback){        Map<String, String> infos = new HashMap<>();        infos.put("id",cardInfo.getId());        infos.put("key",cardInfo.getKey());        service.checkIdCard(infos).enqueue(callback);    }

接下来我们再继续观察一下日志:

07-16 23:25:18.385 19518-19555/? E/MainActivity: log: --> POST http://api.avatardata.cn/IdCard/LookUp http/1.107-16 23:25:18.385 19518-19555/? E/MainActivity: log: Content-Type: application/x-www-form-urlencoded07-16 23:25:18.386 19518-19555/? E/MainActivity: log: Content-Length: 5807-16 23:25:18.387 19518-19555/? E/MainActivity: log: 07-16 23:25:18.387 19518-19555/? E/MainActivity: log: id=510921199103210310&key=af43ed19e5524c55a7bdb33d7f16bc5f07-16 23:25:18.387 19518-19555/? E/MainActivity: log: --> END POST (58-byte body)07-16 23:25:18.819 19518-19555/? E/MainActivity: log: <-- 200 OK http://api.avatardata.cn/IdCard/LookUp (432ms)07-16 23:25:18.819 19518-19555/? E/MainActivity: log: Cache-Control: private07-16 23:25:18.819 19518-19555/? E/MainActivity: log: Content-Length: 11007-16 23:25:18.820 19518-19555/? E/MainActivity: log: Content-Type: application/json; charset=utf-807-16 23:25:18.820 19518-19555/? E/MainActivity: log: Server: Microsoft-IIS/7.507-16 23:25:18.820 19518-19555/? E/MainActivity: log: X-AspNet-Version: 4.0.3031907-16 23:25:18.820 19518-19555/? E/MainActivity: log: X-Powered-By: ASP.NET07-16 23:25:18.820 19518-19555/? E/MainActivity: log: Date: Sun, 16 Jul 2017 15:25:19 GMT07-16 23:25:18.820 19518-19555/? E/MainActivity: log: OkHttp-Sent-Millis: 150021871847807-16 23:25:18.820 19518-19555/? E/MainActivity: log: OkHttp-Received-Millis: 150021871881807-16 23:25:18.821 19518-19555/? E/MainActivity: log: 07-16 23:25:18.821 19518-19555/? E/MainActivity: log: {"result":{"address":"四川省蓬溪县","sex":"M","birthday":"1991-03-21"},"error_code":0,"reason":"Succes"}07-16 23:25:18.821 19518-19555/? E/MainActivity: log: <-- END HTTP (110-byte body)07-16 23:25:18.830 19518-19518/? E/MainActivity: onResponse: 0

OK,明天可以给领导交差了!!!

原创粉丝点击