Android WebView预渲染介绍

目录
  • 前言
  • 术语对齐
  • 客户端可以从哪些方面优化h5页面的加载速度?
    • 优化思路
  • 预渲染的基本实现逻辑是怎样的?
    • 预创建
    • 预创建个数
    • 预创建时机
    • 预创建复用
    • 预渲染
      • 预渲染时机
      • 预渲染有效性校验
      • 时间有效性
      • 状态有效性
      • 页面显示状态通知
      • 其它注意事项
  • 预渲染存在哪些局限性?
  • 总结

前言

在一个Hybrid项目中,必不可少的就是加载h5页面。h5页面的加载性能极大影响着用户体验,并会从各方面影响到我们APP的业务数据。试想,假设一个h5页面要花好几秒才能打开,那用户还会使用我们的APP吗?所以今天我们讲一讲,客户端上优化h5页面加载速度的一种方式:预渲染

照例,抛出本篇文章要解决的几个问题:

  • 客户端可以从哪些方面优化h5页面的加载速度?
  • 预渲染的基本实现逻辑是怎样的?
  • 预渲染存在哪些局限性?

术语对齐

术语 描述
WebView 用于承载h5页面的native组件。
预创建 在用户打开一个h5页面之前,在内存中先创建好一个WebView实例。当用户打开h5页面时,可以直接取到预先创建好的WebView实例,用于承载h5页面。
预渲染 在用户打开一个h5页面之前,在内存中不仅预创建好了WebView实例,还进一步根据url提前渲染好了WebView。当用户打开指定的url页面时,可以直接拿预先渲染好了的WebView进行展示。

客户端可以从哪些方面优化h5页面的加载速度?

我们可以看一下,在Android上完整打开一个WebView需要经历怎样的一个链路(来自优秀的前辈们):

所以,要想优化WebView的加载速度,就要想办法去缩短链路中这些节点所耗费的时间

从客户端的角度上来讲,Page初始化就是H5容器的初始化。一般而言,容器初始化时间是比较快的(如果没有太多初始化逻辑的话),优化空间有限。WebView的初始化相对就比较复杂,涉及到了浏览器内核在主线程的初始化。APP冷启动后,首次创建WebView时需要去初始化浏览器内核。

这里要分3种情况去看:

  • 全新安装APP,冷启动后首次打开一个WebView,耗时最长,可能会需要1000ms左右,取决于浏览器内核。
  • 非全新安装APP,冷启动后首次打开一个WebView,耗时分布在500ms左右。
  • 冷启动后,非首次打开一个WebView,耗时就非常短了,分布在15ms左右。

这里我们可以看到,第1种和第2种情况是存在比较大的提升空间的

当我们创建好WebView,执行WebView#loadUrl()时,WebView就会经历上图中的白屏-loading-可交互状态。这几个阶段,可以说是整个链路中耗时占比最大的一部分。但客户端在这里能做的优化是很有限的,而且通常需要跟前端、服务端去配合优化,比如可以并行请求数据,这里就先不发散了。但如果换个角度思考,客户端先不去干涉后面这几个阶段的逻辑,而是提前去执行后面这几个阶段的逻辑,那么不也就相当于提高加载速度了么?这其实就是我们要讲的预加载。

优化思路

所以我们的优化思路主要针对WebView初始化阶段,以及WebView加载阶段。

通过预创建WebView,去解决首次创建WebView耗时长的问题。

通过预渲染WebView,去提前经历用户需要等待的白屏-loading阶段,当用户打开相应页面时,能够直接上屏展示,给用户的感觉就是秒开。

预渲染的基本实现逻辑是怎样的?

预创建

预创建是预渲染的前提(没有预创建好怎么预渲染呢..),所以我们先讲下预创建。预创建WebView,一个基本原则就是,当内存中没有预创建的WebView可以复用(即预创建没有命中)时,就走原来创建WebView的逻辑。

预创建个数

这里我们选择只预创建1个WebView。之所以选择1个,是因为我们预创建WebView的根本目的,是为了解决APP首次安装/冷启动时,第一个WebView加载慢的问题。后续的WebView实例的创建都是很快的。所以,即使后面没有命中预创建的WebView,用的重新创建的WebView,也就是多花了15ms左右的时间,影响是很小的。所以综合下来,预创建1个WebView的性价比是最高的,多了反而浪费内存。

预创建时机

这里的时机要分为三个,第一个时机是在冷启动后,我们需要进行预创建。可以选择把这个时机放到进入首页后,用IdleHandler进行主线程闲时创建。当然也可以选择前置。前置的话有可能会影响到APP的启动,所以如果不是特别有必要的话,建议还是后置一些。

第二个时机是在预创建的WebView被拿去复用后,此时也是需要预创建的。因为一旦被拿去复用,意味着我们缓存中已经没有可用的WebView了,若一个pha页面又打开了另外一个pha页面,我们在这个case最好也能提供预创建的WebView。

以上两个时机都是自动触发。后来发现一个场景,当用户在某个路径比较深的页面时,若需要预加载下一个页面,那么这个页面往往是不需要一冷启动就预渲染的。这时候就需要一个接口让业务方能在用户打开页面之前将该页面进行预加载。

void preload(Context context, String url);

预创建复用

复用WebView需要注意一点,每个WebView都是跟指定的Context绑定的,但预创建时,还获取不到WebView未来要绑定的Context。因此预创建时可以用MutableContextWrapper包ApplicationContext去创建。MutableContextWrapper支持我们将其中的BaseContext进行替换,复用预创建的WebView时,将ApplicationContext替换为需要绑定的Context即可。同时根据“预创建时机”中说的,在复用时,往栈顶插入一个新的预创建的WebView。相应的,当页面关闭时,我们也需要将绑定的Context解绑,防止内存泄漏。

大致的逻辑如下图所示:

这里也贴出部分伪代码:

/**
 * 闲时预创建
 */
private void preCreateWebView() {
    Looper.myQueue().addIdleHandler(new MessageQueue.IdleHandler() {
        @Override
        public boolean queueIdle() {
            // 预创建webView
            WebView webView = new WebView(new MutableContextWrapper(APPLICATION_CONTEXT));
            cache.add(webView);
            return false;
        }
    });
}

/**
 * 获取预创建的WebView
 * @param context 要与使用的webview绑定的context
 * @return
 */
public PHAWebView acquirePreCreateWebView(Context context) {
    WebView webview;
    // 缓存中无可用WebView时,直接新建
    if (cache.isEmpty()) {
        webview = createWebView();
    } else {
        webview = cache.peekWebView();
    }
    // 更改context
    if (webview != null && webview.getContext() instanceof MutableContextWrapper) {
        MutableContextWrapper webViewMutableContext = (MutableContextWrapper) webview.getContext();
        webViewMutableContext.setBaseContext(context);
    }
    return webview;
}

预渲染

预渲染,其实就是在预创建的基础上,执行WebView#loadUrl(),将页面提前渲染完成。但这里面还有一些细节点需要关注。

预渲染时机

预渲染的时机也是分为两个,一个是冷启动后的主线程闲时阶段进行预渲染,这一点跟预加载的时机保持一致。还有一个我们可以选择在页面关闭时进行预渲染。打比方说,假设我预渲染了页面A,那么用户在访问完页面A后,我需要再次预渲染页面A,从而保证页面A的实效性。

预渲染白名单

首先,预渲染是有对象的,预渲染的对象就是页面url。而预创建是没有特定对象的,只需要随时准备一个可用的WebView就行了,谁都可以用。但预渲染不行,不告诉我预渲染谁,我还怎么预渲染。

所以,从初始化开始,我们就已经决定好了该预渲染哪些页面,也就是预渲染白名单。白名单可通过服务器配置的方式进行下发。但也不能一股脑把页面全都配上,因为内存是有限的,配太多可能会引起低端机型的OOM。

预渲染有效性校验

所谓的有效性校验,就是在复用预渲染WebView时,校验这个WebView是否被正常预渲染了。如果失效,就走预创建/重新创建的逻辑。这里分两个角度来校验WebView的有效性:时间有效性与状态有效性。

时间有效性

存在这样一种场景:当我们已经预渲染了A页面,且用户一直没有访问。某个时刻这个页面做了更新,即重新发布了,那么如果用户这时候去访问A页面,看到的还是旧的A页面。所以这里需要在预渲染页面时,给页面设置一个过期时间,若复用预渲染WebView时已经过期了,就说明WebView已经失效了,需要重新loadUrl保证页面的实效性。

状态有效性

状态有效性就是去校验预渲染的WebView最终是否有渲染成功,这一点我们可以通过WebViewClient的生命周期回调(onPageFinished/onReceivedError)来进行判断。之所以要做这个校验,是为了防止一开始预渲染就失败了,却还是拿这个WebView去进行展示。比如,假设我们在网络异常状态下去进行了预渲染,在网络恢复正常后用户访问预渲染页面,若不进行状态校验,那么看到的就会是网络异常状态下的WebView了。

页面显示状态通知

页面显示状态,通俗来讲就是页面现在是离屏的,还是上屏的。在h5的一些业务场景中,有一部分是需要感知到页面的显示状态的。比如引导类的动画,比如会场的一些倒计时等等。所以我们需要将页面的显示状态同步到h5那边。

实现上,就是要在预渲染WebView时给h5注入一个全局的环境变量,window.page_on_screen=false。当复用WebView,即上屏时,再将window.page_on_screen设置为true,同时发一个通知给h5。这样h5就可以根据同步到的显示状态来控制自己的业务逻辑。

其它注意事项

预渲染、预创建,本质上是用空间换时间的优化,所以是比较耗费内存的。所以我们需要在内存不足的时候,及时将内存中待使用的WebView给回收掉,避免APP发生OOM。

另外,因为预渲染离屏加载了页面,所以页面的初始化行为是需要纳入评估的,只有评估通过后,才能放入预渲染白名单中。具体的初始化行为包括但不限于:业务的曝光埋点、前端逻辑(如倒计时、跨天活动)、消费型(如首次引导)、后端流量评估、页面在后台是否会有声音、是否会弹框(系统框、权限框、对话框...)等等。

预渲染存在哪些局限性?

  • 低端机内存空间有限,预渲染白名单的实际配置数量需要视情况进行调整。
  • 预渲染页面必须经过白名单配置。页面url、参数发生改变,配置也需要改变。这一点其实也是有优化空间的,即离屏预渲染时加载url前缀域名,上屏时再根据完整的url参数做逻辑调整。实现上会比较麻烦,可以视ROI情况进行投入。
  • 预渲染页面的实效性无法保证。预渲染页面一旦重新部署,端上是不能立刻感知到并重新加载的。按上面的预渲染时机,目前只有以下二个场景会触发端上对预渲染页面的更新:
    • 1.冷启动;
    • 2.页面被访问后关闭;
    • 3.业务调用接口主动注入。

所以大家如果有比较好的方案欢迎分享给我呀!

  • 命中率。预渲染页面是不能百分百命中的,即即使我们把某个页面配置进了预渲染白名单,app也有可能没预渲染上这个页面。有很多异常场景会影响到命中率,比如:

    • 1.上面讲到的预渲染的时间有效性与状态有效性;
    • 2.服务端下发的预渲染白名单没有及时拉取到;
    • 3.主线程一直繁忙,导致预渲染逻辑一直没执行;
    • 4.内存不够,将缓存的预渲染WebView回收掉了。

总结

所以虽然预渲染能从表面上实现h5页面的秒开,但也不是万能的,是存在一些缺陷的(否则也不需要别的优化手段了)。但我认为是诸多优化手段中比较简单却又能立竿见影的一个手段,特别是对本身h5页面加载就非常慢的app而言。所以如果还没做起来的同学可以试一试,后面再结合其它优化的手段抹除不足。今天就讲到这里啦。

到此这篇关于Android WebView预渲染介绍的文章就介绍到这了,更多相关Android WebView预渲染内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Android WebView实现全屏播放视频

    目录 介绍 主要代码 介绍 最近项目开发中用到了WebView播放视频的功能,总结了开发中犯过的错误,这些错误在开发是及容易遇到的,所以我这里总结了一下,希望大家看到后不要再犯类似的错误,尽可能提高开发效率: 这个Demo我这里也参考了网上写的一个比较好的一个Demo,经过总结修改,写出来的. 主要代码 以下是相应代码: MainActivity: package com.androidwebviewdemo; import android.app.Activity; import androi

  • Android WebView输入框被档问题升级解析

    目录 前言 issue 5497问题 布局顶起后距离软键盘有一定距离 Activity的webview输入框被挡 总结 前言 之前写过一篇文章,有讲如何处理Android输入框被软键盘挡住的问题,无论是原生的还是webview的,这里面主要的问题是webview的问题比较难处理,没有看过的可以先看看  Android 输入框被挡问题完美解决方案 主要是issue 5497这个问题.然后可能有些朋友觉得,这篇文章没能解决他的问题,或者说按照我的代码去写又会出现新的问题.这说明没有能理解这篇文章写的

  • Android WebView缓存机制优化加载慢问题

    目录 一 .前言 二 .WebView存在的性能问题 2.1 H5 页面加载速度慢 2.1.1 渲染速度慢 2.1.2 页面资源加载缓慢 2.2 耗费流量 2.3 总结 三 .解决方案 3.1 前端H5的缓存机制 3.1.1 缓存机制 3.1.2 缓存模式 3.2 资源预加载 3.2.1 预加载WebView对象 3.2.2 预加载H5资源 3.3 自身构建缓存 3.3.1 需求场景 3.3.2 实现步骤 3.3.3 具体实现 3.3.4 具体实例 一 .前言 由于H5具备 开发周期短.灵活性好

  • Android WebView控件基本使用示例

    Android WebView用于在 android 中显示网页.可以从相同的应用程序或 URL 加载网页.它用于在 android 活动中显示在线内容. Android WebView 使用 webkit 引擎来显示网页. android.webkit.WebView 是 AbsoluteLayout 类的子类. Android WebView 类的loadUrl()和loadData()方法用于加载和显示网页. Android WebView 示例 让我们看看使用 Web 视图显示 baid

  • Android WebView基础应用详解

    目录 一.WebView的基础配置 二.WebView支持播放音乐 三.WebView支持视频播放 四.WebChromeClient 五.WebViewClient 1.重定向问题 2.实现预加载 3.增加错误页面展示限制 4.解决页面白屏问题 附GitHub源码:WebViewExplore 一.WebView的基础配置 WebSettings ws = getSettings(); ws.setBuiltInZoomControls(true);// 隐藏缩放按钮 ws.setLayout

  • Android webView加载数据时内存溢出问题及解决

    目录 Android webView加载数据时内存溢出 Android内存问题 (内存溢出 内存泄漏 内存抖动) 总结 Android webView加载数据时内存溢出 今天使用webView加载数据时   如果数据太长就会崩溃,造成内存溢出,在网上查找了一下资料之后   终于把它解决了,谨在此记录 1.不要在XML里面写webView 可以使用一个占位布局 <FrameLayout     android:id="@+id/layoutWebView"     android:

  • Android WebView如何判断是否滚动到底部

    目录 场景: 分析: 实现: 总结 场景: 我们有时候需要弹一些必读公告,但是呢可能会要去你看完之后才可以关掉,所以就需要滚动到底部才显示关闭按钮,而公告什么的往往又是基于富文本的,那么在展示在Android上时就要用到WebView,基于这个要求就有了判断 WebView 判断是否滚动到底部. 分析: 要判断是否到底部那么我们先来分析有哪几种情况,当html文档加载到WebView后会有一下两种情况. WebView里的html内容没有填充满,就是无滚动条情况.html内容的高度比WebVie

  • 浅谈Android开发Webview的Loading使用效果

    目录 前言 1. loading的选择 2. loading显示时机的问题 3. 体验优化 4. loading最终设计效果 总结 前言 在开发webview的loading效果的时候会有一些问题,这边记录一些碰到的常见的问题,并且设计出一套Loading的方案来解决相关的问题. 1. loading的选择 开发loading效果的原因在于webview加载页面的时候,有时候会耗时,导致不显示内容又没有任何提示,效果不太好,所以需要在webview使用的地方加上loading的效果,其实更好的体

  • Android WebView软键盘遮挡输入框方案详解

    目录 背景 纪实 方案 实现 总结 背景 笔者在使用 WebView 加载含有输入框的 H5 页面时,点击输入框后,输入框会被软键盘遮挡住,无法看到输入的内容,这很影响用户体验. 笔者想着这种业务场景比较常见,遂上网搜索一番,果不其然,有不少同志遇到这个问题,想来这个问题很好解决了.笔者一一尝试了同志们提供的解决方案,结果要不是没有作用,要不是效果不太满意,只好自己另辟蹊径了. 注:在笔者的业务场景中,App是全屏的,即没有顶部的系统栏,也没有底部的导航栏,所以笔者的解决方案,可能不适用于其他场

  • Android TextView预渲染研究

    Android中的TextView是整个framework中最复杂的控件之一,负责Android中显示文本的大部分工作,framwork中的许多控件也直接或者间接的继承于TextView,例如Button,EditText等.其内部实现也相当复杂,单论代码行数来说,android-22中TextView有足足9509行,另外,TextView中许多操作都非常繁重,例如setText操作,需要设置SpanWatcher,或者需要重现创建一个SpannableString,还需要根据情况重新创建Te

  • 详解服务端预渲染之Nuxt(介绍篇)

    现在前端开发一般都是前后端分离,mvvm和mvc的开发框架,如Angular.React和Vue等,虽然写框架能够使我们快速的完成开发,但是由于前后台分离,给项目SEO带来很大的不便,搜索引擎在检索的时候是在网页中爬取数据,由于单页面应用读取到的页面是几乎空白的,无法爬取到任何数据信息. <!DOCTYPE html> <html> <head> <meta charset=utf-8> <meta name=viewport content=&quo

  • Android webview加载H5方法详细介绍

    目录 1,安卓APP 怎么用webview加载H5 2,H5怎么调用安卓定义的方法 3,安卓怎么调用H5定义的方法 这篇文章主要阐述3个知识点 安卓APP 怎么用webview加载H5 H5怎么调用安卓定义的方法 安卓怎么调用H5定义的方法 1,安卓APP 怎么用webview加载H5 安卓端定义个webview xml 页面,代码如下所示: <?xml version="1.0" encoding="utf-8"?> <WebView xmlns

  • Android WebView使用方法详解 附js交互调用方法

    目前很多Android app都内置了可以显示web页面的界面,会发现这个界面一般都是由一个叫做WebView的组件渲染出来的,学习该组件可以为你的app开发提升扩展性. 先说下WebView的一些优点: --可以直接显示和渲染web页面,直接显示网页 --webview可以直接用html文件(网络上或本地assets中)作布局 --和JavaScript交互调用 一.基本使用 首先layout中即为一个基本的简单控件: <WebView android:id="@+id/webView1

  • Android WebView实现长按保存图片及长按识别二维码功能

    先来简单说一下本文所要实现的功能:用户在浏览网页的时候,长按某一区域,识别如果是图片,则弹出弹框,出现保存图片的功能.同时识别图片是否是二维码,如果是则在弹框中追加识别二维码功能. 细节上:保存图片的弹框要显示在手指长按的位置:选择图片保存后,可以让用户直接去相册查看:选择识别二维码,判断是是不是网址,是的话可以让用户选择复制或访问,否则可以让用户选择复制或搜索. 然后再来看一下效果图: 保存图片 save.gif 识别包含普通文字的二维码: text.gif 识别包含网址的二维码: code.

  • Android WebView与JS交互全面详解(小结)

    Android 和 H5 都是移动开发应用的非常广泛.市面上很多App都是使用Android开发的,但使用Android来开发一些比较复杂附属类,提示性的页面是得不偿失的.而H5具有开发速度快,更新不用依赖于App的更新,只需要服务端更新相应的页面即可,所以,App和H5页面相结合就显得尤为重要.而android和H5都不可能每次都是独立存在的,而是相互影响也相互的调用,获取信息等,例如,H5页面要获取App中的用户的基本信息,或者App端要操作H5页面等,下面来看看这两是怎么交互的 目录 1.

  • Flutter WebView 预加载实现方法(Http Server)

    目录 背景 分析 HttpServer 接下来? 资源配置 下载解压与本地存储 版本管理与更新 获取LocalServer Url并加载Webview 兜底措施 统一管理 展示与分析 总结 Demo 背景 WebView是在APP中,可以很方便的展示web页面,并且与web交互APP的数据.方便,并且更新内容无需APP发布新版本,只需要将最新的web代码部署完成,用户重新刷新即可. 在WebView中,经常能够听到的一个需求就是:减少首次白屏时间,加快加载速度.因为加载web页面,必然会受到网络

  • Android WebView的使用方法及与JS 相互调用

    Android WebView的使用方法及与JS 相互调用 1.添加网络权限 <uses-permission android:name="android.permission.INTERNET" /> 2.WebSettings 对访问页面进行设置. WebView mWebView = new WebView(this); WebSettings webSettings = mWebView .getSettings();//支持获取手势焦点,输入用户名.密码或其他 m

  • Android WebView线性进度条实例详解

    推荐阅读:Android Webview添加网页加载进度条实例详解 先给大家展示下效果图:这个效果图大家一看就懂,在生活经常见到 1.wevbview_progressbar.xml <layer-list xmlns:android="http://schemas.android.com/apk/res/android" > <!-- 背景 --> <item android:id="@android:id/background"&g

随机推荐