前端资源预加载并展示进度条

  我们经常会看到,一些站点在首次进入的时候会先显示一个进度条,等资源加载完毕后再呈现页面,大概像这样:

  

  然后整个页面的操作就会非常流畅,因为之后没必要再等待加载资源了。尤其是在移动端,或者是页游中,这样做能避免页面出现白屏(等待加载图片),很大程度提升用户体验。那这种技术是如何实现的呢?其实非常简单,本文就来从基础细节探究一番。

为什么需要资源预加载

大多时候,我们的页面并不是一次渲染完毕的,而是随着用户的操作,不断修改DOM节点,如果你动态插入了一个图片节点,那么浏览器要马上发一个http请求,把图片加载下来然后渲染在页面上,如果用户此时的网速不佳,那么加载这张图片可能就会消耗几秒钟时间,此时页面上什么都没有(白屏)。最坏的情况,如果你的应用图片很多,半天加载不出几张图,用户很可能就在白屏的那几秒跳走了。在游戏中更严重,主角的图片如果加载不出来,让用户玩空气去?

除了在DOM中插入图片节点,其他凡是涉及到要展示一张新图片的操作,浏览器都得即时去请求图片。比如,为某个节点添加如下css类来增加背景图片:

.bg1{
     background: url(http://p2.qhimg.com/t01ed1438874f940dc0.jpg);
}

  或者是动态修改了src属性、在canvas绘制图片等,这些都会即时请求新资源。

那么,资源预加载为什么能解决上述问题呢?

我们预加载的资源,浏览器会缓存下来,再次使用的时候,浏览器会检查是不是已经在缓存中,如果在,则直接用缓存的资源,不发送请求,或者由服务端返回304 not modified(304只带请求头信息,不传输资源)。这样使用一张图片的时间会大大缩减,我们的页面看起来会非常流畅,妈妈再也不用担心用户会跳走了~

也就是说,预加载的资源我们并不需要手动保存,由浏览器自动放到缓存了。

资源预加载的场景

什么样的项目需要预加载资源呢?

范围应该锁定单页面应用,SPA的视图一般都是一步一步来呈现的,各种资源通过异步请求来获取,为了追求原生app般的流畅体验,可以把一些资源预加载下来。当然对于一些业务相关的图片资源,也可考虑延迟加载,但延迟加载不是本文讨论的范畴。

视图/图片较多的专题页面,或者是需要逐帧图片来完成的动画效果,最好都要做预加载。

HTML5游戏,图片一般都比较多,而且很多逐帧动画,必须要预加载,事实上一些游戏引擎都会提供相应功能。

哪些资源需要预加载呢?

web中包含的资源有很多种,图片、音视频之类的媒体文件,另外就是js、css文件,这些都需要发送请求来获取。那这些资源难道我们都预加载?

当然不是了,预加载也是需要消耗时间的,总不能让用户等你加载个几十秒钟吧。具体预加载哪些资源,需要基于具体的考虑,也跟你的项目相关。以下是一些我的想法:

js、css文件不需进行预加载。现在写js基本都用requirejs之类的加载器,而且最后都会进行压缩合并,将请求数降到最低,最终只有一个文件,有些团队甚至还将压缩后的代码直接放在行内,这样一个多余的请求都没有了。

那么需要预加载的就是媒体文件了,图片、音视频之类。这类资源也得根据实际情况来选择哪些需要预加载。比如大多数页面装饰性图片就需要预加载,而由业务动态获取的图片则无法预加载(预先不知道地址)。用作音效、小动画的音视频可以预加载,一个半小时长的视频就不能预加载了。

预加载的原理与加载进度的获取

上面都是纸上谈兵的一些观点,下面我们该从技术的角度来思考一下预加载该如何实现。

原理其实也相当简单,就是维护一个资源列表,挨个去加载列表中的资源,然后在每个资源加载完成的回调函数中更新进度即可。

以图片为例,大致的代码应该是这样:

var image = new Image();
image.onload = function(){};
image.onerror = function(){};
image.src = url;

这样就OK啦,图片已经进缓存,留着以后使用吧。

再说进度,这个进度严格来讲并不是文件加载的实时进度,因为我们只能在每个文件加载完成的时候执行回调,无法像timeline中那样拿到文件加载的实时进度。

计算方法就很简单了,当前加载完的资源个数/资源总数*100,就是加载进度的百分比了。

资源预加载小插件:resLoader.js介绍

本文的重点终于来了。。。额

根据上面的原理,我写了一个插件,用来做资源预加载。

具备的特征如下:

1. 自定义资源列表,用于预加载

2. 自定义onProgress,想展示成进度条还是百分比数字还是个性的设计都可

3. 开始和结束可配置回调函数

4. 只支持图片的预加载

5. 支持amd、cmd加载器加载,同时支持直接用<script>标签引入使用

6. 不依赖其他库

用法如下:

var loader = new resLoader({
     resources : [
          ‘http://p2.qhimg.com/t01ed1438874f940dc0.jpg‘,
          ‘http://p9.qhimg.com/t01b4ff03b72c7dc6c7.jpg‘,
          ‘http://p2.qhimg.com/t01dd90dfbec92074d0.jpg‘,
          ‘http://p7.qhimg.com/t01cfec6d87cde457c5.jpg‘,
          ‘http://p9.qhimg.com/t01943ced462da67833.jpg‘,
          ‘http://p0.qhimg.com/t01943ced462da67833.jpg‘,
          ‘http://p6.qhimg.com/t01aa15a7ba7ccb49a7.jpg‘,
          ‘http://p8.qhimg.com/t010f1e8badf1134376.jpg‘,
          ‘http://p8.qhimg.com/t01cf37ea915533a032.jpg‘,
          ‘http://p3.qhimg.com/t0193d8a3963e1803e9.jpg‘,
          ‘http://p3.qhimg.com/t01cd6a4d4b4bd4457b.jpg‘
     ],
     onStart : function(total){
          console.log(‘start:‘+total);
     },
     onProgress : function(current, total){
          console.log(current+‘/‘+total);
          var percent = current/total*100;
          $(‘.progressbar‘).css(‘width‘, percent+‘%‘);
          $(‘.progresstext .current‘).text(current);
          $(‘.progresstext .total‘).text(total);
     },
     onComplete : function(total){
          alert(‘加载完毕:‘+total+‘个资源‘);
     }
});

loader.start();

  各项参数都直接明了,不再多说了。在上面的例子中,我自己定义onProgress函数,做了一个简单的进度条,你也可以做其他实现。函数为你传入了current和total,分别表示当前完成的资源个数和资源总个数,可用于计算进度。

  效果可看在线demo:点击这里

  另外附上下载地址,感兴趣的朋友可以拿去使用:http://files.cnblogs.com/files/lvdabao/resLoader.zip

再多说两句,关于xhr2新特性

前边的废话貌似有点多。。。想直接用插件的下载下去用就好,有问题在此留言讨论。

这里想多说的东西是关于加载进度的,我上面说了我们只能获取到的是进度其实是离散的点,不是连续的。其实利用HTML5的xhr2的新特性我们也可以尝试获取更加精确的进度。因为xhr2新增了一个非常有趣的特性:可以从服务端获取文件数据。我们以前从服务端返回的数据都是字符串,现在可以直接返回Blob类型的文件。那么在这里做一个猜想,能不能利用此特性,做更加精确的进度计算呢?我在此处只是提出一种可能性,还未做实践。我们知道xhr2新增的upload属性可以让我们获取到文件上传的进度信息,但对于返回的数据,却无法直接提供进度信息,所以要想依靠它来实现还得做其他工作。

时间: 2024-10-10 14:29:39

前端资源预加载并展示进度条的相关文章

前端性能优化 - 资源预加载

转帖: https://css-tricks.com/prefetching-preloading-prebrowsing 当提到前端性能优化时,我们首先会联想到文件的合并.压缩,文件缓存和开启服务器端的 gzip 压缩等,这使得页面加载更快,用户可以尽快使用我们的 Web 应用来达到他们的目标. 资源预加载是另一个性能优化技术,我们可以使用该技术来预先告知浏览器某些资源可能在将来会被使用到. 引用 Patrick Hamann 的解释: 预加载是浏览器对将来可能被使用资源的一种暗示,一些资源可

前端性能优化 – 资源预加载

导语    当提到前端性能优化时,我们首先会联想到文件的合并.压缩,文件缓存和开启服务器端的 gzip 压缩等,这使得页面加载更快,用户可以尽快使用我们的 Web 应用来达到他们的目标. 资源预加载是另一个性能 当提到前端性能优化时,我们首先会联想到文件的合并.压缩,文件缓存和开启服务器端的 gzip 压缩等,这使得页面加载更快,用户可以尽快使用我们的 Web 应用来达到他们的目标. 资源预加载是另一个性能优化技术,我们可以使用该技术来预先告知浏览器某些资源可能在将来会被使用到. 引用 Patr

手把手教你构建 Android WebView 的缓存机制 &amp; 资源预加载方案

前言 由于H5具备 开发周期短.灵活性好 的特点,所以现在 Android App大多嵌入了 Android Webview 组件进行 Hybrid 开发 但我知道你一定在烦恼 Android Webview 的性能问题,特别突出的是:加载速度慢 & 消耗流量 今天,我将针对 Android Webview 的性能问题,提出一些有效解决方案. 目录 1. Android WebView 存在什么性能问题? Android WebView 里 H5 页面加载速度慢 耗费流量 下面会详细介绍. 1.

页面资源预加载(Link prefetch)

不管是浏览器的开发者还是普通web应用的开发者,他们都在做一个共同的努力:让Web浏览有更快的速度感觉.有很多已知的技术都可以让你的网站速度变得更快:使用CSS sprites,使用图片优化工具,使用.htaccess设置页面头信息和缓存时间,JavaScript压缩,使用CDN等.我曾经介绍过本站上使用的一些速度优化技术.而在HTML5里,出现了一个新的用来优化网站速度的新功能:页面资源预加载/预读取(Link prefetch). 页面资源预加载/预读取(Link prefetch)是什么?

基于用户行为的图片等资源预加载

一.图片的懒加载和预加载 懒加载和本文要提到的预加载实际是不同的概念. 典型的懒加载,例如本博文章的图片,当用户滚动图片进入窗体的时候,才去加载:或者用户点击选项卡,原本隐藏的图片此时再去加载,这个也称之为懒加载. 而预加载则是,用户还没有行为发生,资源已经加载完毕,从这一定义来讲,我们传统图片啪啪啪全部加载完毕,本质上也是预加载,好处就在于,体验好啊,没有泛白或者菊花的出现.不足也很明显,那就是资源可能白白加载了,尤其视频之类的,小明打开bilibili就是来围观广告的,结果,百兆视频巴拉拉魔

HTML5页面资源预加载(Link prefetch)功能加速页面加载速度

页面资源预加载/预读取(Link prefetch)是什么?来自MDN的解释: 页面资源预加载(Link prefetch)是浏览器提供的一个技巧,目的是让浏览器在空闲时间下载或预读取一些文档资源,用户在将来将会访问这些资源.一个Web页面可以对浏览器设置一系列的预加载指示,当浏览器加载完当前页面后,它会在后台静悄悄的加载指定的文档,并把它们存储在缓存里.当用户访问到这些预加载的文档后,浏览器能快速的从缓存里提取给用户. 简单说来就是:让浏览器预先加载用户访问当前页后极有可能访问的其他资源(页面

[转]资源预加载

资源预加载可以提升用户体验,如果每次用户打开页面都要加载图片,js,css等资源,会影响用户体验.资源预加载可以一定程度上改善这种情况. 我们可以做的是,但第一个页面load完的时候,在用户阅读网页的空隙,把下一个页面所用的资源提前加载过来cache住,这样下个页面就直接读缓存资源了,这样可以一定程度改善用户体验. 那么预加载资源需要解决的主要问题是JS加载过来不会被直接执行,css加载过来不会更改页面样式. 这样就会产生很多方案, 这里介绍一种不错的兼容方案: 1. IE下用new Image

关于cocos2d-x 与 cocos2d-html5 资源预加载的思考

移动端资源预加载,可以做到需要加载的时候,从本地磁盘加载到内存,当纹理不需要的时候,都是强制清理内存里的纹理占用: cc.TextureCache.getInstance().removeAllTextures(); cc.TextureCache.getInstance().dumpCachedTextureInfo();//test 打印仍然在使用的纹理 cc.TextureCache.purgeSharedTextureCache(); cc.SpriteFrameCache.getIns

cordova加载层、进度条、文件选择插件

在做cordova项目的时候,感觉应用的响应速度跟原生应用比相差甚远,一个主要问题就是如加载层.进度条等弹出对话框的效率不行.毕竟项目中的这些弹框都是用dom拼成的,dom的渲染效率和原生控件比起来慢也是正常的.为了能够让应用尽可能的接近原生应用的效果,我业余时间做了一套对话框插件,使用后应用的使用体验立马有了很大的提升.兼容ios和安卓平台.插件包括加载层.进度条.文件选择.项目已经放到了github,有兴趣的小伙伴可以下载试一试. 插件只支持cordova5.0及以上版本,初始化如下: co