升级okhttp3.0+和retrofit2.0+的过程以及遇到的坑

包名变化

okhttp3.0之前是:com.squareup.okhttp.*,

而到3.0之后变成:okhttp3.*

retrofit2.0之前:retrofit.*

retrofit2.0之后:retrofit2.*

都在末尾增加了版本号,显示的更专业,但是这也给我们升级带来了麻烦。

首先看一下retrofit1.9+okhttp2.2的配置代码

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. public class MainRetrofit {  
  2.   
  3.     final MeoHttp mService;  
  4.     final Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'").serializeNulls().create();  
  5.     RequestInterceptor requestInterceptor = new RequestInterceptor() {  
  6.         @Override  
  7.         public void intercept(RequestFacade request) {  
  8.             request.addHeader("User-Agent", AppConstants.USER_AGENT);  
  9.             request.addHeader("X-Requested-With""XMLHttpRequest");  
  10.             request.addHeader("Cookie", AppConstants.PHPSESSID);  
  11.   
  12.         }  
  13.     };  
  14.     MainRetrofit() {  
  15.         OkHttpClient client = new OkHttpClient();  
  16.         client.setReadTimeout(41, TimeUnit.SECONDS);  
  17.         RestAdapter restAdapter = new RestAdapter.Builder()  
  18.                 .setClient(new OkClient(client))  
  19.                 .setEndpoint(MainFactory.HOST)  
  20.                 .setConverter(new GsonConverter(gson))  
  21.                 .setLogLevel(RestAdapter.LogLevel.NONE)  
  22.                 .setRequestInterceptor(requestInterceptor)  
  23.                 .build();  
  24.         mService = restAdapter.create(MeoHttp.class);  
  25.      
  26.     }  
  27.   
  28.     public MeoHttp getService(){  
  29.         return mService;  
  30.     }  
  31.   
  32. }  
okhttp3.0以及retrofit2.0均采用了builder构建者模式配置
[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. final RequestApi api;  
  2.    final Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'").serializeNulls().create();  
  3.    Interceptor requestInterceptor;  
  4.    private static final okhttp3.OkHttpClient.Builder httpClient = new OkHttpClient.Builder();  
  5.    public MainRetrofit(final boolean hasCookie) {  
  6.        httpClient.readTimeout(60, TimeUnit.SECONDS).  
  7.                connectTimeout(2, TimeUnit.SECONDS).build();  
  8.        requestInterceptor = new Interceptor() {  
  9.            @Override  
  10.            public Response intercept(Chain chain) throws IOException {  
  11.                Request request = chain.request();  
  12.                String cookie = "";  
  13.                if (hasCookie) {  
  14.                    cookie = AppConfig.Net.COOKIE_HEAD + CookieManager.getCookie();  
  15.                }  
  16.                Request compressedRequest = request.newBuilder()  
  17.                        .header("Cookie", cookie)  
  18.                        .header("Accept-Language", Locale.getDefault().toString())  
  19.                        .header("Accept-Charset""utf-8")  
  20.                        .header("Connection""Keep-Alive")  
  21.                        .header("User-Agent", getUserAgent())  
  22.                        .build();  
  23.                Response response = chain.proceed(compressedRequest);  
  24.                return response;  
  25.            }  
  26.        };  
  27.        httpClient.interceptors().add(requestInterceptor);  
  28.        OkHttpClient okHttpClient = httpClient.build();  
  29.        Retrofit client = new Retrofit.Builder()  
  30.                .baseUrl(AppConfig.Net.HOST)  
  31.                .addConverterFactory(new ScalarsConverterFactory())  
  32.                .addConverterFactory(GsonConverterFactory.create(gson))  
  33.                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())  
  34.                .client(okHttpClient)  
  35.                .build();  
  36.        //retrofit 没有日志打印功能  
  37.        /*if (AppConfig.IS_DEVELOPING) { 
  38.            //日志可用 
  39.            client.setLogLevel(RestAdapter.LogLevel.FULL); 
  40.        } else { 
  41.            client.setLogLevel(RestAdapter.LogLevel.NONE); 
  42.        }*/  
  43.        api = client.create(RequestApi.class);  
而且retrofit2.0没有打印日志的功能,所以只能在okhttp的拦截器里打印日志

retrofit1.9的接口调用方式:

GET:

/**
 * 获得用户地址
 */
@GET("/user/Address?")
void getAddress(Callback<AddressInfo>callback);

POST:

/**
 * 上传画作
 * @param content 图集描述
 * @param atk token值
 * @param file 文件
 */
@Multipart
@POST("/user/uploadPicture?")
void upLoadPaint(@Part("content")String content, @Part("t")String atk,
                 @Part("upfile") TypedFile file, Callback<BaseResponse> callback);
而对应2.0之后的方式是这样的

GET:

/**
 * 获得用户地址
 */
@GET("/user/Address?")
Calll<AddressInfo> getAddress();
POST:

/**
 * 上传画作
 * @param content 图集描述
 * @param atk token值
 * @param file 文件
 */
@Multipart
@POST("/user/uploadPicture?")
Call<BaseResponse> upLoadPaint(@Part("content")String content, @Part("t")String atk,
@Part("upfile") RequestBody body);
大致的升级过程算是完成了

接下来讲讲有什么坑:

1.纯粹使用retrofit接口返回不能是void ,而是相对应了Call<T>,接口的实现不在是直接使用Call回调,而是用getAddress().enqueue(new CallBack)

  这里面的变动有点大,如果没做好回调函数的封装,那升级的工作量就有点大了,这样返回一个call可以用于取消请求。

2.在Retrofit 1.9中,如果获取的 response 不能背解析成定义好的对象,则会调用failure。但是在Retrofit 2.0中,不管 response 是否能被解析。onResponse总是会被调用。但是在结果不能被解析的情况下,response.body()会返回null。别忘了处理这种情况。

如果response存在什么问题,比如404什么的,onResponse也会被调用。你可以从response.errorBody().string()中获取错误信息的主体。
3.retrofit2.0去掉了一些类:RetrofitError,TypeFile MultipartTypedOutput,如果在1.9中用到,升级的时候需要注意

4.@Path注解在1.9的时候使用方式:
@Path(value="url", encode=false) String url

2.0时会导致encoded失效,此时推荐使用@Url动态替换url地址

@Multipart
@POST
Call<String>  upload(@Url String url,
            @PartMap Map<String, RequestBody> params);
5.多图片上次的坑: MultipartTypedOutput,以及TypeFile在2.0之后被去除了

1.9的多图片上传方式:

/**
 * @param multipartTypedOutput 多个参数加载
 */
@POST("/user/applyPainting?")
void request2BePaint(@Body MultipartTypedOutput multipartTypedOutput,Callback<BaseResponse> callback);
multipartTypedOutput组装:

MultipartTypedOutput multipartTypedOutput = new MultipartTypedOutput();
multipartTypedOutput.addPart("t",new TypedString(painter.atk));
multipartTypedOutput.addPart("content",new TypedString(painter.content));
for (int i=0;i<painter.fileImages.length;i++){
   multipartTypedOutput.addPart("upfile["+i+"]",new TypedFile("image/jpeg", new File(painter.fileImages[i])));
}
相对于简单些,而到2.0之后就感觉很繁琐

/**
 * 上传文件
 */
@Multipart
@POST
Call<String>  upload(@Url String url,
            @PartMap Map<String, RequestBody> params);
组装map以及RequestBody

RequestBody requestFile = null;
Map<String, RequestBody>  photos = new HashMap<>();
for (int i = 0; i < formFiles.size(); i++) {
    
    //生成临时压缩图片文件
    final File tempFile = FileUtils.createFile(AppConfig.MAIN_DIR_IMG,
            "IMG_TEMP_" + System.currentTimeMillis() + ".jpg");
    Bitmap bitmap = PhotoUtils.compressBitmap(formFiles.get(i).getFile().getPath());
    FileOutputStream fos = new FileOutputStream(tempFile);
    bitmap.compress(Bitmap.CompressFormat.JPEG, quality, fos);
    requestFile = RequestBody.create(MediaType.parse("multipart/form-data"), tempFile);
    photos.put("AttachmentKey"+i+"\"; filename=\""+keys.get(i),requestFile);}
以上就是我升级的过程,如有问题和错误请留言
### 回答1: OkHttp、RxJava和Retrofit是个非常常用的组合,用来进行网络请求和处理。 首先,OkHttp是一个开源的HTTP客户端,提供了简洁的接口,用于与服务器进行通信,并且支持HTTP/2协议,拥有连接池、请求重试和缓存等功能。 然后,RxJava是一个基于观察者模式的异步编程库,可以用于处理异步操作,如网络请求、文件IO等。它的核心是Observable(被观察者)和Subscriber(订阅者),通过各种操作符可以对数据进行变换和处理。 最后,Retrofit是一个RESTful风格的HTTP请求库,它基于OkHttp,使用了Retrofit的注解和接口定义的方式,可以方便地进行网络请求。它支持动态代理,可以将网络请求接口转化为对应的HTTP请求,支持同步和异步请求,并且可以将响应数据转化为Java对象。 综上所述,我们可以使用OkHttp作为底层网络库,然后结合RxJava和Retrofit进行网络请求和数据处理。使用Retrofit的注解和接口定义方式,可以简化网络请求的代码,并且通过RxJava的操作符可以对请求结果进行变换和处理,使得代码更加清晰和可读性。 在使用过程中,可以先创建一个Retrofit实例,并指定OkHttpClient作为网络客户端,然后定义一个接口,在该接口中使用Retrofit的注解,定义网络请求的方法。最后,通过创建该接口的实例,即可进行网络请求,并结合RxJava进行操作。 总之,使用OkHttp、RxJava和Retrofit组合进行网络请求可以提高效率和可读性,并且可以处理各种复杂的网络场景,是一种非常实用的方式。 ### 回答2: OKHTTP、RXJava和Retrofit是Android开发中常用的三个库,可以一起使用来进行网络请求和数据处理。 OKHTTP是一个用于处理网络请求的库,可以发送HTTP请求并获取服务器返回的数据。它提供了简洁的API和高效的网络堆栈,可以很好地处理网络请求。我们可以使用OKHTTP来发送SOAP请求到WebService,并获得响应。 RXJava是一个流编程库,它提供了一种被观察者和观察者模式,可以简化异步操作和事件处理。我们可以使用RXJava来处理OKHTTP返回的响应数据,在主线程或后台线程中进行处理,实现数据的异步处理和流式编程。 Retrofit是一个基于OKHTTP的RESTful风格的网络请求库,它提供了一种简洁的方式来定义和发送HTTP请求,并将响应转换为可用的Java对象。我们可以使用Retrofit来定义WebService接口,然后使用注解来指定请求方法、路径和参数,Retrofit会自动帮我们处理请求和响应。 通过OKHTTP的原生支持、RXJava的异步处理和Retrofit的网络请求,我们可以很方便地使用OKHTTP、RXJava和Retrofit一起发送WebService请求。首先,我们可以使用Retrofit定义WebService接口,再使用RXJava来处理OKHTTP返回的响应数据,实现简洁高效的网络请求和数据处理。 综上所述,OKHTTP、RXJava和Retrofit是Android开发中常用的网络请求库,它们能够很好地协作,实现对WebService的请求和响应的处理。通过使用它们,我们可以简化网络请求的编写,并实现高效的数据处理。 ### 回答3: OkHttp、RxJava和Retrofit是三个在Android开发中常用的网络请求库,它们在一起能够提供更加便捷和高效的网络请求处理。 首先,OkHttp是一个开源的HTTP客户端,它能够处理网络请求、连接管理、请求重试等一系列的网络相关事务。它的特点是简单易用、性能优越、可定制性强。我们可以通过使用OkHttp来发送和接收基于HTTP的请求响应,并进行网络请求的管理和处理。 其次,RxJava是一个响应式编程框架,它基于观察者模式和函数式编程的思想,提供了一系列强大的操作符和线程切换的能力。我们可以使用RxJava来处理异步任务,加快网络请求的响应时间,并且提供方便的线程切换和错误处理机制。 最后,Retrofit是一个RESTful风格的网络请求框架,它结合了OkHttp和RxJava的强大功能。它提供了一种简单的方式来定义和处理RESTful API的请求和响应。我们可以使用Retrofit来创建和处理webservice的请求,根据API的接口定义来发送请求,并将返回的结果映射到Java对象中。 综上所述,使用OkHttp、RxJava和Retrofit能够方便地进行webservice的网络请求,并在处理过程中提供更好的性能和便利性。这三个库的结合能够大大简化网络请求的开发工作,提高开发效率,并提供更好的用户体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值