本文主要介绍Android网络请求中的一些基础知识及缓存的应用
目录
http请求&响应相关知识
okhttp&Interceptor
retrofit+ okhttp + rxjava的缓存实现
http相关知识
直接贴上jude的图,jude是我的好老师
请求包.png
例子.png
响应包.png
例子.png
请求头描述了客户端向服务器发送请求时使用的http协议类型,所使用的编码,以及发送内容的长度,等等。
相应的响应头用来描述服务器回给你对所返回的content的一些描述,服务器类型,我返回的编码,我返回的内容有多长等等。
okhttp
okhttp是现在开发中相对成熟的网络请求框架,对比HttpURLConnection有更易用的API和更优良的性能。
在这里对于常规的get/post 同步/异步的方法就不提了,主要是对应上面的介绍写一些高级用法。
Request&Response的header部分
这部分为了下文缓存部分如何添加header做介绍
请求包对应okhttp中的Request,响应包对应Response
当写请求头的时候,使用header(name, value)可以设置唯一的name、value。如果已经有值,旧的将被移除,然后添加新的。使用addHeader(name, value)
可以添加多值(添加,不移除已有的)。当读取响应头时,使用header(name)
返回最后出现的name、value。通常情况这也是唯一的name、value。如果没有值,那么header(name)将返回null。如果想读取字段对应的所有值,使用headers(name)会返回一个list。
Interceptor
Interceptor,就像他的翻译一样“拦截器”,拦截你的Request,做一些想做的事情再发送出去。
改写请求和响应
Interceptor interceptor =newInterceptor() {publicResponseintercept(Chain chain)throwsIOException{returnnull; } };
可以通过以下代码获取被拦截的Request
Request request = chain.request();
然后可以对这个request进行newBuilder() 重新产生一个request发送出去
调用以下代码产生一个满足请求的响应
Response response = chain.proceed(request);
当我们执行到proceed,就会去判断是否有拦截器,有的话先执行拦截器里的intercept,而在intercept里一般会进行一些自定义操作并且调用procced去判断是否要继续执行拦截器操作还是直接去获取网络请求
publicResponseproceed(Request request)throwsIOException{//判断还有拦截器需要执行不,生成新的ApplicationInterceptorChain并调用它的intercept去执行用户定义的操作if(this.index < Call.this.client.interceptors().size()) { Call.ApplicationInterceptorChain chain = Call.this.new ApplicationInterceptorChain(this.index +1, request,this.forWebSocket);return((Interceptor)Call.this.client.interceptors().get(this.index)).intercept(chain); }else{returnCall.this.getResponse(request,this.forWebSocket); } }
缓存(retrofit+okhttp+rxjava)
app网络数据的离线缓存实现有很多种办法,例如存进数据库(保存json使用时拿出来解析),存专有文件,或者SharedPreference等等,也可以自己实现LruCache和
DiskLruCache这两种缓存策略构成二级缓存(内存和磁盘)
缓存对于移动端是非常重要的存在。
减少请求次数,减小服务器压力.
本地数据读取速度更快,让页面不会空白几百毫秒。
在无网络的情况下提供数据。
okhttp的缓存设计和浏览器的缓存设计差不多,可以通过添加响应头的形式进行缓存处理。
关于添加响应头和其他缓存知识请看这个链接web缓存相关概念
(笔者还没看完 = =)
retrofit是依赖okhttp的一套RESTful架构的Android(Java)客户端实现
通过构造retrofit时的.client()方法更改其中的okhttp的实现,从而达到缓存的效果,在这里不介绍retrofit的具体用法。
下面是一个例子实现了有网缓存,无网络只读取缓存
File cacheFile =newFile(APP.getContext().getExternalCacheDir(),"ZhiBookCache"); Cache cache =newCache(cacheFile,1024*1024*50); Interceptor interceptor =newInterceptor() {@OverridepublicResponseintercept(Chain chain)throwsIOException{ Request request = chain.request();if(!HttpUtils.isNetworkConnected(APP.getContext())) { request = request.newBuilder() .cacheControl(CacheControl.FORCE_CACHE) .build(); } Response response = chain.proceed(request);if(HttpUtils.isNetworkConnected(APP.getContext())) {intmaxAge =0*60;// 有网络时 设置缓存超时时间0个小时response.newBuilder() .header("Cache-Control","public, max-age="+ maxAge) .removeHeader("Pragma")// 清除头信息,因为服务器如果不支持,会返回一些干扰信息,不清除下面无法生效.build(); }else{// 无网络时,设置超时为4周intmaxStale =60*60*24*28; response.newBuilder() .header("Cache-Control","public, only-if-cached, max-stale="+ maxStale) .removeHeader("Pragma") .build(); }returnresponse; } }; client =newOkHttpClient.Builder().cache(cache) .addInterceptor(interceptor) .build(); retrofit =newRetrofit.Builder() .baseUrl(RetrofitAPI.BASIC_DAILY) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) .client(client) .build(); retrofitAPI = retrofit.create(RetrofitAPI.class);
之后调用下面代码获取请求后的序列化对象
retrofitAPI.getDaily() .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(newObserver() {@OverridepublicvoidonCompleted(){ }@OverridepublicvoidonError(Throwable e){ Log.d(TAG,e.getMessage()); }@OverridepublicvoidonNext(DailyBean bean){ listener.onGetDailySuccess(bean); } });
最后附上retrofit接口部分
@GET("news/latest")ObservablegetDaily();
参考文章
Android网络请求心路历程
Retrofit 源码解读之离线缓存策略的实现
OkHttp源码解析
OkHttp使用进阶
作者:srtianxia
链接:https://www.jianshu.com/p/42a396430be5
來源:
著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。