Android中关于Volley的使用(五)缓存机制的深入认识

1)Volley可以在SD卡中缓存图片,那可不可以在SD卡中缓存Json数据呢?

2)如果断网了,Volley是不是就不能用了,存在SD卡中的数据是不是就用不了了?

在Volley中,默认使用的缓存实现是 DiskBasedCache,在创建RequestQueue的时候,同时也会创建一个DiskBasedCache对象,如下:

[java]  view plain copy
  1. RequestQueue queue = new RequestQueue(new DiskBasedCache(cacheDir), network);  

而缓存文件的位置就是由CacheDir提供的,它的值如下:

Android中关于Volley的使用(五)缓存机制的深入认识_第1张图片

在前面的VolleyDemo中,我们加载了一些图片,也加载了一些json数据,那么我们现在就来看看在缓存中的数据,有没有缓存图片的数据,同时,有没有缓存json的数据。

Volley文件夹下面有以下几条数据:

Android中关于Volley的使用(五)缓存机制的深入认识_第2张图片

在文件夹中,以"-993813455"开头的文件,其实就是图片文件的缓存,而以-165747开头的那个文件,其实就是我们从天气网站拿下来的json数据的缓存,我们可以打开看一下里面文件的信息,下面是缓存文件-993813455-446463727的内容:

[html]  view plain copy
  1.  <       http://img.my.csdn.net/uploads/201403/03/1393854084_6138.jpg       "FhHq7DMn1v_jbLzbG84-iD_stqzT"榟?E  橮奓  橮奓            X-ViaN       1.1 gzck20:8107 (Cdn Cache Server V2.0), 1.1 dgck149:0 (Cdn Cache Server V2.0)       ETag       "FhHq7DMn1v_jbLzbG84-iD_stqzT"       X-Reqid       JAQAAFDYivywH2IT       X-Log       MC;IO:2       Content-Length       16713       Content-Transfer-Encoding       binary  
  2.        Connection  
  3.        keep-alive       Server        nginx/1.4.4  
  4.        Cache-Control       public, max-age=31536000       X-Whom       nb5       Date       Sat, 05 Apr 2014 16:01:19 GMT       Access-Control-Allow-Origin       *       Content-Disposition&       inline; filename="1393854084_6138.jpg"       X-Android-Received-Millis  
  5.        1396713679439        Content-Type  
  6.        image/jpeg  
  7.        Accept-Ranges       bytes       X-Android-Sent-Millis  
  8.        1396713678988??JFIF   d d  ?Ducky     <  ?慼ttp://ns.adobe.com/xap/1.0/ <?xpacket begin="锘? id="W5M0MpCehiHzreSzNTczkc9d"?>  
  9. <x:xmpmeta xmlns:x="adobe:ns:meta/" x:xmptk="Adobe XMP Core 4.1-c036 46.276720, Mon Feb 19 2007 22:13:43        ">  
  10.  <rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">  
  11.   <rdf:Description rdf:about=""  
  12.     xmlns:dc="http://purl.org/dc/elements/1.1/"  
  13.     xmlns:xapRights="http://ns.adobe.com/xap/1.0/rights/"  
  14.    xapRights:Marked="Romain Guy"  
  15.    xapRights:WebStatement="Romain Guy">  
  16.    <dc:rights>  
  17.     <rdf:Alt>  
  18.      <rdf:li xml:lang="x-default">Romain Guy</rdf:li>  
  19.     </rdf:Alt>  
  20.    </dc:rights>  
  21.   </rdf:Description>  
  22.  </rdf:RDF>  
  23. </x:xmpmeta>  

我们可以看到文件的开头就是要展示的图片的url地址,再来看看Json数据的内容:

[html]  view plain copy
  1.  0       http://www.weather.com.cn/data/sk/101280101.html        (&?E                            Age       297       Transfer-Encoding       chunked       Date       Sat, 05 Apr 2014 16:31:37 GMT       X-Android-Received-Millis  
  2.        1396715892761        Content-Type       text/html; charset=utf-8  
  3.        Connection       close       Server        Apache/2.2.0       X-Android-Sent-Millis  
  4.        1396715892611{"weatherinfo":{"city":"骞垮窞","cityid":"101280101","temp":"18","WD":"瑗垮寳椋?,"WS":"1绾?,"SD":"92%","WSE":"1","time":"00:20","isRadar":"1","Radar":"JC_RADAR_AZ9200_JB"}}  

不仅可以看到json数据的ulr地址,我们还可以看到下面具体的json数据,如“weatherinfo...”等信息。

从上面这两条数据,我们可以回答那位朋友的第一个问题,SD卡中不仅仅缓存图片,也是有缓存Json数据的。而事实上,通过网络获取来的数据其实都是字节流,Volley只是通过Response的Header信息来设置缓存记录的生命周期等,具体代码如下:

[java]  view plain copy
  1. public static Cache.Entry parseCacheHeaders(NetworkResponse response) {  
  2.     long now = System.currentTimeMillis();  
  3.   
  4.     Map<String, String> headers = response.headers;  
  5.   
  6.     long serverDate = 0;  
  7.     long serverExpires = 0;  
  8.     long softExpire = 0;  
  9.     long maxAge = 0;  
  10.     boolean hasCacheControl = false;  
  11.   
  12.     String serverEtag = null;  
  13.     String headerValue;  
  14.   
  15.     headerValue = headers.get("Date");  
  16.     if (headerValue != null) {  
  17.         serverDate = parseDateAsEpoch(headerValue);  
  18.     }  
  19.   
  20.     headerValue = headers.get("Cache-Control");  
  21.     if (headerValue != null) {  
  22.         Log.v(Helper.TAG, "Has Cache-Control");  
  23.         hasCacheControl = true;  
  24.         String[] tokens = headerValue.split(",");  
  25.         for (int i = 0; i < tokens.length; i++) {  
  26.             String token = tokens[i].trim();  
  27.             if (token.equals("no-cache") || token.equals("no-store")) {  
  28.                 return null;  
  29.             } else if (token.startsWith("max-age=")) {  
  30.                 try {  
  31.                     maxAge = Long.parseLong(token.substring(8));  
  32.                     Log.v(Helper.TAG, "Max AGe = " + maxAge);  
  33.                 } catch (Exception e) {  
  34.                 }  
  35.             } else if (token.equals("must-revalidate") || token.equals("proxy-revalidate")) {  
  36.                 maxAge = 0;  
  37.             }  
  38.         }  
  39.     }  
  40.   
  41.     headerValue = headers.get("Expires");  
  42.     if (headerValue != null) {  
  43.         serverExpires = parseDateAsEpoch(headerValue);  
  44.     }  
  45.   
  46.     serverEtag = headers.get("ETag");  
  47.   
  48.     // Cache-Control takes precedence over an Expires header, even if both exist and Expires  
  49.     // is more restrictive.  
  50.     if (hasCacheControl) {  
  51.         softExpire = now + maxAge * 1000;  
  52.     } else if (serverDate > 0 && serverExpires >= serverDate) {  
  53.         // Default semantic for Expire header in HTTP specification is softExpire.  
  54.         softExpire = now + (serverExpires - serverDate);  
  55.     }  
  56.   
  57.     Log.v(Helper.TAG, "SoftExpire = " + softExpire);  
  58.       
  59.     Cache.Entry entry = new Cache.Entry();  
  60.     entry.data = response.data;  
  61.     entry.etag = serverEtag;  
  62.     entry.softTtl = softExpire;  
  63.     entry.ttl = entry.softTtl;  
  64.     entry.serverDate = serverDate;  
  65.     entry.responseHeaders = headers;  
  66.   
  67.     return entry;  
  68. }  

所以,其实不管数据是图片,还是Json数据,对于缓存来说,这并不重要。

那么我们来看看第二个问题,断网了,Volley是不是就不能用了?

通过前面Volley的框架图,我们知道,Volley首先会去缓存中找数据,如果找不到才会去网络中获取数据,所以如果缓存中有数据的话,它就不用去跟网络打交道了,那么跟断不断网其实也就关系不大了。

把手机设置成飞行模式,然后我们再去加载图片,我们可以看到在Log中的信息,如下:

[java]  view plain copy
  1. 04-06 11:44:43.671: D/Volley(6541): [1] MarkerLog.finish: (+0   ) [ 1] add-to-queue  
  2. 04-06 11:44:43.671: D/Volley(6541): [1] MarkerLog.finish: (+434 ) [1059] cache-queue-take  
  3. 04-06 11:44:43.671: D/Volley(6541): [1] MarkerLog.finish: (+17  ) [1059] cache-hit  
  4. 04-06 11:44:43.671: D/Volley(6541): [1] MarkerLog.finish: (+9   ) [1059] cache-hit-parsed  
  5. 04-06 11:44:43.671: D/Volley(6541): [1] MarkerLog.finish: (+1   ) [1059] post-response  
  6. 04-06 11:44:43.681: D/Volley(6541): [1] MarkerLog.finish: (+0   ) [ 1] done  
  7. 04-06 11:44:43.691: V/com.lms.volleydemo(6541): Has Cache-Control  
  8. 04-06 11:44:43.691: V/com.lms.volleydemo(6541): Max AGe = 31536000  
  9. 04-06 11:44:43.691: V/com.lms.volleydemo(6541): SoftExpire = 1428291883693  
  10. 04-06 11:44:43.691: V/com.lms.volleydemo(6541): Completely unexpired cache hit. http://img.my.csdn.net/uploads/201403/03/1393854094_4652.jpg  
可以看到Volley的确是可以从缓存中拿出图片数据来展示的,相信对于这个问题的答案,那位朋友应该也了解了吧。

那么我们再来看看Json数据的log吧,如下:

[java]  view plain copy
  1. 04-06 12:18:37.521: V/com.lms.volleydemo(6541): cache-hit-expired for request http://www.weather.com.cn/data/sk/101280101.html  
  2. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (99   ms) [ ] http://www.weather.com.cn/data/sk/101280101.html 0x88751869 NORMAL 27  
  3. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+0   ) [ 1] add-to-queue  
  4. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+1   ) [1059] cache-queue-take  
  5. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+74  ) [1059] cache-hit-expired  
  6. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+1   ) [1062] network-queue-take  
  7. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+20  ) [1062] post-error  
  8. 04-06 12:18:37.551: D/Volley(6541): [1] MarkerLog.finish: (+3   ) [ 1] done  
  9. 04-06 12:18:38.683: D/memalloc(6541): ion: Mapped buffer base:0x5c2a9000 size:3768320 offset:0 fd:51  
在飞行模式下,我们看到它首先是cache_hit_expired,然后才会去网络获取数据,但是因为没有连网,所以就会出现错误,它就获取不到数据了。

那么Volley中是如何判断expired的数据的呢,如下:

[java]  view plain copy
  1. /** True if the entry is expired. */  
  2. public boolean isExpired() {  
  3.     return this.ttl < System.currentTimeMillis();  
  4. }  

而ttl的值就是上面parseCacheHeader中的softExpire的值了,这个值是由缓存中的MaxAge的值来计算的,但其实所有这些都是从Response中取出来的值。

对应于我们demo中的天气预报的url,通过网络传回来的response是没有cache-control的,所以它的maxAge也就是0了,从而导致ttl的值也一直是0,所以就一直是过期的,需要去从网张中获取。

我觉得这是因为天气是瞬息万变的,搞缓存是没有意义的,而对于其它的json数据,其实取决于服务器传回来的Header信息中CacheControl的信息。

当然,如果真想自己缓存,又不想它过期,可以把Volley的源代码也导入项目中,修改源代码也是可以的,不过这就。。。。

你可能感兴趣的:(Android中关于Volley的使用(五)缓存机制的深入认识)