weex 在 iOS 上如何实现常见的网络缓存

简介: weex 旨在兼顾web动态性与native的用户体验,如果想将两者的优势最大化,那么缓存就显得格外重要,本文介绍如何利用缓存,实现weex页面迅速打开,甚至“秒开”的效果。

作者:阿里云-移动云-大前端团队

前言

weex 旨在兼顾web动态性与native的用户体验,如果想将两者的优势最大化,那么缓存就显得格外重要,本文介绍如何利用缓存,实现weex页面迅速打开,甚至“秒开”的效果。

正文

要实现 native 端的缓存,需要两个层面:

  • JS 文件缓存
  • request 请求缓存

单纯缓存JS文件是没用的,除非你的JS文件是hello world级别:不会在JS内部进行网络请求加载其他资源。有人说我的JS也有网络请求,请求了一张图片,也是可以的呀?十有八九那是 SDWebImage 功劳,那是你实现了图片加载的协议,SDWebImage已经帮你做了缓存了。

下面详细来看下如何让 weex 在 iOS 上支持常见的网络缓存:

JS 文件缓存

大致有两种思路:

  • 预加载类型:在启动后,客户端主动到服务端拉取会用到JS并缓存。这样下次用到该JS文件事即可实现“秒开”。
  • 类似于传统的网络缓存类型:第一次加载该JS文件时,需要通过网络加载,下次访问时就可以不走网络,需要设置cache策略。

预加载方式也是常见的缓存方式,启动后预先加载,在此不做赘述。

第二种类型,有一篇文章已经有比较详细的论述:

具体的思路用流程图表示如下:

具体的步骤如下:

  • 下载JS前重写renderByUrl
  • 渲染时重写render
  • 在页面使用自定义WXSDKInstance替换原WXSDKInstance

第一步 下载JS前重写renderByUrl

在拿到服务端JS的url后,首先判断是否有本地JS缓存,若有则对比本地JS及服务端JS的MD5进行校验,校验通过则直接使用本地JS,否则按原计划下载服务端JS。Weex支持使用本地JS文件。

@Override
public void renderByUrl(String pageName, String url, Map<String, Object> options, String jsonInitData, WXRenderStrategy flag) {

   String local = "";
   if(TextUtils.isEmpty(url) || md5Check(url)){
       local = getLocalJs();//获取本地JS路径
   }

   if(!TextUtils.isEmpty(local){
       super.renderByUrl(pageName, local, options, jsonInitData, flag);
   }else {
       super.renderByUrl(pageName, url, options, jsonInitData, flag);
   }
}
/**
* 获取本地JS路径
*/
private String getLocalJs(){
   try {
       File f = new File(context.getFilesDir(), "local_js.txt");
       if(f.exists()) {
           return "file://" + f.getAbsolutePath();
       }
   } catch (Exception e) {
   }
   return "";
}

第二步 渲染时重写render

JS文件获取成功后,若是从服务端下载的JS,则需要进行文件缓存。

@Override
public void render(String pageName, String template, Map<String, Object> options, String jsonInitData, WXRenderStrategy flag) {
   saveWeexFile(template);
   super.render(pageName, template, options, jsonInitData, flag);
}
/**
* 异步存储JS Bundle RX实现
* @param template
*/
private void saveWeexFile(String template){
   if(isLocalFile(getBundleUrl())){
       return;
   }
   Observable
           .just(template)
           .map(new Function<String, Boolean>() {
               @Override
               public Boolean apply(String s) throws Exception {
                   //weex对文件名不敏感,存txt文件
                   return FileUtil.saveFile(context.getFilesDir().getAbsolutePath(), "local_js.txt", s.getBytes("UTF-8"));
                   }
               })
           .subscribeOn(Schedulers.io())
           .subscribe(new Consumer<Boolean>() {
               @Override
               public void accept(Boolean aBoolean) throws Exception {
                   if(aBoolean){
                       //缓存成功
                   }
               }
           },new Consumer<Throwable>(){
               @Override
               public void accept(Throwable throwable) throws Exception {
                   throwable.printStackTrace();
               }
           });
   }
/**
* 异步存储JS Bundle
* @param template
*/
private void saveWeexFile(String template){
   if(isLocalFile(getBundleUrl())){
       return;
   }
   Thread thread = new Thread( new Runnable(){
       @Override
       public void run(){
           FileUtil.saveFile(context.getFilesDir().getAbsolutePath(), "local_js.txt", template.getBytes("UTF-8"));
       }
   });
   thread.start();
}
/**
* 判断是本地文件还是网络url
*/
private boolean isLocalFile(String url){
   if(TextUtils.isEmpty(url){
       return false;
   }
   Uri uri = Uri.parse(url);
   if (uri != null && TextUtils.equals(uri.getScheme(), "file")) {
       return true;
   }
   return false;
}

第三步 在页面使用自定义WXSDKInstance替换原WXSDKInstance

注意:如果weex页面更新不频繁,就没必要每次都进行文件校验。每次启动app只进行一次文件校验并缓存MD5,后续打开页面进行本地MD5校验。

request 缓存

上面介绍的是JS的缓存,但是 JS 文件缓存后,还是无法实现无网络状况下,直接打开 JS 页面,JS 页面还有大量的资源文件,JS 文件内部还是会发送网络请求,这些网络请求依然需要用到缓存策略。

这个部分的缓存基本的思路如下:

  • 与传统的缓存是一样的。
  • 添加一个缓存方式:先缓存后网络。

weex的网络请求部分,可以设置扩展,设置后,所有的weex SDK的网络请求都会经由该扩展处理,所以request部分的请求,实际上与传统的缓存是一样的。比如我们熟悉的NSCache、YYCache等第三方的网络请求方式也是可以复用的。

weex 网络请求扩展部分的代码如下:

《weex官网文档-iOS扩展》

参考链接

相关文章
|
2月前
|
存储 缓存 UED
缓存策略与Apollo:优化网络请求性能
缓存策略与Apollo:优化网络请求性能
|
4天前
|
存储 缓存 安全
基于iOS平台的高效图片缓存策略实现
【4月更文挑战第22天】 在移动应用开发中,图片资源的加载与缓存是影响用户体验的重要因素之一。尤其对于iOS平台,由于设备存储空间的限制以及用户对流畅性的高要求,设计一种合理的图片缓存策略显得尤为关键。本文将探讨在iOS环境下,如何通过使用先进的图片缓存技术,包括内存缓存、磁盘缓存以及网络请求的优化,来提高应用的性能和响应速度。我们将重点分析多级缓存机制的设计与实现,并对可能出现的问题及其解决方案进行讨论。
|
4天前
|
存储 缓存 算法
实现iOS平台的高效图片缓存策略
【4月更文挑战第22天】在移动应用开发中,图片资源的处理是影响用户体验的重要因素之一。特别是对于图像资源密集型的iOS应用,如何有效地缓存图片以减少内存占用和提升加载速度,是开发者们面临的关键挑战。本文将探讨一种针对iOS平台的图片缓存策略,该策略通过结合内存缓存与磁盘缓存的机制,并采用先进的图片解码和异步加载技术,旨在实现快速加载的同时,保持应用的内存效率。
|
14天前
|
存储 缓存 自动驾驶
缓存策略与Apollo:优化网络请求性能
缓存策略与Apollo:优化网络请求性能
|
17天前
|
存储 缓存 iOS开发
基于iOS的高效图片缓存策略实现
【4月更文挑战第9天】在移动应用开发中,图片资源的加载与缓存是影响用户体验的重要因素之一。特别是对于iOS平台,合理设计图片缓存策略不仅能够提升用户浏览图片时的流畅度,还能有效降低应用程序的内存压力。本文将介绍一种针对iOS环境优化的图片缓存技术,该技术通过多级缓存机制和内存管理策略,实现了图片快速加载与低内存消耗的目标。我们将从系统架构、关键技术细节以及性能评估等方面展开讨论,为开发者提供一套实用的图片缓存解决方案。
17 0
|
22天前
|
存储 缓存 iOS开发
实现iOS平台的高效图片缓存策略
【4月更文挑战第4天】在移动应用开发中,图片资源的加载与缓存是影响用户体验的关键因素之一。尤其对于iOS平台,由于设备存储和内存资源的限制,设计一个高效的图片缓存机制尤为重要。本文将深入探讨在iOS环境下,如何通过技术手段实现图片的高效加载与缓存,包括内存缓存、磁盘缓存以及网络层面的优化,旨在为用户提供流畅且稳定的图片浏览体验。
|
7月前
|
缓存 人工智能 JavaScript
「网络」1.你所不知道的浏览器缓存精品答案
「网络」1.你所不知道的浏览器缓存精品答案
|
7月前
|
人工智能 JavaScript Java
0. 专栏介绍
0. 专栏介绍

热门文章

最新文章