Retrofit 2.0 使用总结

这天,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.0
compile 'com.squareup.retrofit2:converter-gson:2.0.2'   //retrofit:2.0封装下的gson
compile '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: 1
07-16 21:52:35.838 9174-9195/? E/MainActivity: run1: 1248
07-16 21:52:36.076 9174-9174/? E/MainActivity: onResponse: 1
07-16 21:52:36.076 9174-9174/? E/MainActivity: onResponse: 7B7F25D86267923209A11145A4EBB34
07-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.1
07-16 23:25:18.385 19518-19555/? E/MainActivity: log: Content-Type: application/x-www-form-urlencoded
07-16 23:25:18.386 19518-19555/? E/MainActivity: log: Content-Length: 58
07-16 23:25:18.387 19518-19555/? E/MainActivity: log: 
07-16 23:25:18.387 19518-19555/? E/MainActivity: log: id=510921199103210310&key=af43ed19e5524c55a7bdb33d7f16bc5f
07-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: private
07-16 23:25:18.819 19518-19555/? E/MainActivity: log: Content-Length: 110
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: Content-Type: application/json; charset=utf-8
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: Server: Microsoft-IIS/7.5
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: X-AspNet-Version: 4.0.30319
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: X-Powered-By: ASP.NET
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: Date: Sun, 16 Jul 2017 15:25:19 GMT
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: OkHttp-Sent-Millis: 1500218718478
07-16 23:25:18.820 19518-19555/? E/MainActivity: log: OkHttp-Received-Millis: 1500218718818
07-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,明天可以给领导交差了!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值