【半原创】将js和css文件装入localStorage加速程序执行
首先感谢某某作者写的文章:http://www.jb51.net/article/12793.htm
直接上代码,注意文件名为env.js
原理如下:
一次批量加要加载的文件存入数组,采用Ajax方式异步载入各个文件,然后采用循环方式逐个执行下载下来的Js或者Css文件,如果已经被缓存(localStorage)的则省略下载过程。
由于JS采用的是单线程模式运行,在执行某一个js时会阻塞其它并发的js执行,所以会按顺序执行各个js。在执行完所有的脚本之后,图片会被浏览器接着加载,所以第一次加载速度略慢,后面就会比较快了。在JQuery Mobile 1.4.5+FireFox/微信浏览器下实测效果不错,IE就被省略了,我主要是要在微信浏览器下使用。
//需要引用别的js的时候,就加上如Env.require("cookie.js"),或Env.require("/common/cookie.js"),是用相对路径还是绝对路径就看喜好了。
//Env.require可用在页面模板中,也可用在js文件中,但一定要保证执行时env.js被显式引入。
//多次Env.require同一个js(不管用相对还是绝对),只有第一次会加载,所以不会重复。 //程序最后发行的版本,用于作为缓存键的前缀,快速更新缓存
var envLastVer = '2014_11_17_17_03'; //用于存放通道名称及通信对象的类,这样可以通过不同通道名称来区分不同的通信对象
function HttpRequestObject() {
this.chunnel = null;
this.instance = null;
} //用于获取的脚本或css文件保存对象
function HttpGetObject() {
this.url = null; //要下载的文件路径
this.cache_key = null; //缓存键
this.chunnel = null; //通道名
this.type = null; //类型,js或css
this.is_fill = false; //内容是否被填充
this.is_exec = false; //内容是否已被执行,防止分几大块载入后重复执行
} //通信处理类,可以静态引用其中的方法
var Request = new function () { //通信类的缓存
this.httpRequestCache = new Array(); //创建新的通信对象
this.createInstance = function () {
var instance = null;
if (window.XMLHttpRequest) {
//mozilla
instance = new XMLHttpRequest();
//有些版本的Mozilla浏览器处理服务器返回的未包含XML mime-type头部信息的内容时会出错。
//因此,要确保返回的内容包含text/xml信息
if (instance.overrideMimeType) {
instance.overrideMimeType = "text/xml";
}
}
else if (window.ActiveXObject) {
//IE
var MSXML = ['MSXML2.XMLHTTP.5.0', 'Microsoft.XMLHTTP', 'MSXML2.XMLHTTP.4.0', 'MSXML2.XMLHTTP.3.0', 'MSXML2.XMLHTTP'];
for (var i = 0; i < MSXML.length; i++) {
try {
instance = new ActiveXObject(MSXML[i]);
break;
}
catch (e) {
}
}
}
return instance;
} /**
* 获取一个通信对象
* 若没指定通道名称,则默认通道名为"default"
* 若缓存中不存在需要的通信类,则创建一个,同时放入通信类缓存中
* @param _chunnel:通道名称,若不存在此参数,则默认为"default"
* @return 一个通信对象,其存放于通信类缓存中
*/
this.getInstance = function (_chunnel) {
var instance = null;
var object = null;
if (_chunnel == undefined)//没指定通道名称
{
_chunnel = "default";
}
var getOne = false;
for (var i = 0; i < this.httpRequestCache; i++) {
object = HttpRequestObject(this.httpRequestCache[i]);
if (object.chunnel == _chunnel) {
if (object.instance.readyState == 0 || object.instance.readyState == 4) {
instance = object.instance;
}
getOne = true;
break;
}
}
if (!getOne) //对象不在缓存中,则创建
{
object = new HttpRequestObject();
object.chunnel = _chunnel;
object.instance = this.createInstance();
this.httpRequestCache.push(object);
instance = object.instance;
}
return instance;
} /**
* 客户端向服务端发送请求
* @param _url:请求目的
* @param _data:要发送的数据
* @param _processRequest:用于处理返回结果的函数,其定义可以在别的地方,需要有一个参数,即要处理的通信对象
* @param _chunnel:通道名称,默认为"default"
* @param _asynchronous:是否异步处理,默认为true,即异步处理
* @param _paraObj:相关的参数对象
*/
this.send = function (_url, _data, _processRequest, _chunnel, _asynchronous, _paraObj) {
if (_url.length == 0 || _url.indexOf("?") == 0) {
alert("由于目的为空,请求失败,请检查!");
return;
}
if (_chunnel == undefined || _chunnel == "") {
_chunnel = "default";
}
if (_asynchronous == undefined) {
_asynchronous = true;
}
var instance = this.getInstance(_chunnel);
if (instance == null) {
alert("浏览器不支持ajax,请检查!")
return;
}
if (_asynchronous == true && typeof (_processRequest) == "function") {
instance.onreadystatechange = function () {
if (instance.readyState == 4) // 判断对象状态
{
if (instance.status == 200) // 信息已经成功返回,开始处理信息
{
_processRequest(instance, _paraObj);
}
else {
alert("您所请求的页面有异常,请检查!");
}
}
}
}
//_url加一个时刻改变的参数,防止由于被浏览器缓存后同样的请求不向服务器发送请求
if (_url.indexOf("?") != -1) {
_url += "&requestTime=" + (new Date()).getTime();
}
else {
_url += "?requestTime=" + (new Date()).getTime();
}
if (_data.length == 0) {
instance.open("GET", _url, _asynchronous);
instance.send(null);
}
else {
instance.open("POST", _url, _asynchronous);
instance.setRequestHeader("Content-Length", _data.length);
instance.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
instance.send(_data);
}
if (_asynchronous == false && typeof (_processRequest) == "function") {
_processRequest(instance, _paraObj);
}
}
} var Env = new function () {
this.needLoadObject = new Array(); //获取env.js文件所在路径
this.envPath = null;
this.getPath = function () {
this.envPath = document.location.pathname;
this.envPath = this.envPath.substring(0, this.envPath.lastIndexOf("/") + 1);
var _scripts = document.getElementsByTagName("script");
var _envPath = null;
var _scriptSrc = null;
for (var i = 0; i < _scripts.length; i++) {
_scriptSrc = _scripts[i].getAttribute("src");
if (_scriptSrc && _scriptSrc.indexOf("env.js") != -1) {
break;
}
}
if (_scriptSrc != null) {
if (_scriptSrc.charAt(0) == '/') {
this.envPath = _scriptSrc.substr(0, _scriptSrc.length - 6);
}
else {
this.envPath = this.envPath + _scriptSrc.substr(0, _scriptSrc.length - 6);
}
}
}
this.getPath(); //获取文件后缀名
this.getFileExt = function (fileUrl) {
var d = /\.[^\.]+$/.exec(fileUrl);
return d.toString().toLowerCase();
} //依次放入要载入的文件
this.pushNeedLoad = function (url) {
var _absUrl = null;
if (url.charAt(0) == '/')
_absUrl = url;
else
_absUrl = this.envPath + url; var object = new HttpGetObject();
object.url = _absUrl;
object.cache_key = envLastVer + _absUrl; //利用版本号+绝对路径生成缓存键
object.chunnel = 'ch' + (this.needLoadObject.length + 1);
object.type = this.getFileExt(_absUrl); //尝试从缓存获取
var cacheContent = localStorage.getItem(object.cache_key);
if (cacheContent) { object.is_fill = true; } this.needLoadObject.push(object);
return this;
} //依次装载要处理的文件
this.batchLoad = function () {
for (var i = 0; i < this.needLoadObject.length; i++) {
var item = this.needLoadObject[i];
var processGet = function (_instance, _paraObj) {
localStorage.setItem(_paraObj.cache_key, _instance.responseText); //缓存文件
_paraObj.is_fill = true;
}
if (item.is_fill == false) {
Request.send(item.url, "", processGet, item.chunnel, false, item); //采用同步方式载入
}
}
return this;
} //依次执行要处理的文件
this.batchExec = function () {
var runCss = function (_css) { document.write('<style type="text/css">' + _css + '</style>'); }
var runJs = function (_js) {
if (window.execScript)
window.execScript(_js);
else
window.eval(_js);
}
//依次执行,由于js为单线程执行,每执行一个js都会阻塞其它,所以可以保证顺序执行
for (var i = 0; i < this.needLoadObject.length; i++) {
var item = this.needLoadObject[i];
if (item.is_exec == false) {
if (item.type == '.js') {
runJs(localStorage.getItem(item.cache_key));
item.is_exec = true; //标记已执行,下次不会再执行
}
else if (item.type == '.css') {
runCss(localStorage.getItem(item.cache_key));
item.is_exec = true; //标记已执行,下次不会再执行
}
}
}
}
}
下面是调用方法:
Env.pushNeedLoad("jquery.mobile-1.4.5/jquery.min.js")
.pushNeedLoad("jquery.mobile-1.4.5/jquery.mobile-1.4.5.min.css")
.pushNeedLoad("/plus_in/weixin/procedure/scripts/task.util.js")
.pushNeedLoad("/plus_in/weixin/procedure/scripts/emcp.mobile.js")
.pushNeedLoad("/plus_in/weixin/procedure/scripts/common.index.js")
.pushNeedLoad("jquery.mobile-1.4.5/jquery.mobile-1.4.5.min.js")
.pushNeedLoad("mobiscroll.2.6/css/mobiscroll.custom-2.6.2.min.css")
.pushNeedLoad("mobiscroll.2.6/js/mobiscroll.custom-2.6.2.min.js")
.pushNeedLoad("/plus_in/weixin/procedure/style/base.css")
.batchLoad().batchExec();
通过火狐F12观察,发现上面的脚本第一次会被加载,后面将会直接从localstorage中读取,节省了很多,将js用于微信浏览器后,也节省了很多带宽。不过第一次加载还是有些慢的,毕竟还是有那么多数据。
【半原创】将js和css文件装入localStorage加速程序执行的更多相关文章
- 将js和css文件装入localStorage加速程序执行
原理如下: 一次批量加要加载的文件存入数组,采用Ajax方式异步载入各个文件,然后采用循环方式逐个执行下载下来的Js或者Css文件,如果已经被缓存(localStorage)的则省略下载过程. 由于J ...
- 常用js,css文件统一加载方法,并在加载之后调用回调函数
原创内容,转载请注明出处! 为了方便资源管理和提升工作效率,常用的js和css文件的加载应该放在一个统一文件里面完成,也方便后续的资源维护.所以我用js写了以下方法,存放在“sourceControl ...
- js或css文件后面的参数是什么意思?
经常看到不少导航网站测样式或js文件后面加了一些参数,主要是一你为一些并不经常更新的页面重新加载新修改的文件. 经常遇到页面里加载的js与css文件带有参数,比如: <script type=& ...
- MVC学习随笔----如何在页面中添加JS和CSS文件
http://blog.csdn.net/xxjoy_777/article/details/39050011 1.如何在页面中添加Js和CSS文件. 我们只需要在模板页中添加JS和CSS文件,然后子 ...
- grunt 单独压缩多个js和css文件【转】
原文地址:http://xiaomiya.iteye.com/blog/2177877 使用grunt来压缩前端js,css文件 因为最近做的客户端本地项目有用到十几个js,js提交之前都需要压缩.用 ...
- gulp-rev同时将js和css文件写在一个rev-manifest.json文件里面的方式探讨
参考: https://segmentfault.com/q/1010000002876613 https://github.com/sindresorhus/gulp-rev 测试发现,在官网上最主 ...
- 使用Maven构建Java Web项目时,关于jsp中引入js、css文件路径问题。
今天有点闲,自己动手搭建一个Java Web项目,遇到jsp中引入js.css文件时路径不正确的问题,于是在网上查阅了很多资料,最终都无法解决问题,于是,上stackoverflow找到了解决方法,这 ...
- [转][前端优化]使用Combres合并对js、css文件的请求
本文转自:http://www.cnblogs.com/parry/archive/2011/01/28/Reduce_Http_Request_Using_Combres_For_Js_Css.ht ...
- 为js和css文件自动添加版本号
web应用必然要面对缓存问题,无论前台后台都会涉足缓存.特别是对于前端而言,缓存利用的是否得当直接关系到应用的性能. 通常情况下,我们会倾向于使用缓存,因为缓存一方面可以减少网络开销,一方面可以减轻服 ...
随机推荐
- python练习程序(c100经典例11)
题目: 古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? a=1;b=1 print a,b, for i ...
- 一起学CUDA(零)
1.Nvidia为什么引入CUDA最近实验室已经有不少豪在入手本本了,因为学霸居多,所以大家一般都会说对显卡要求不高,不玩大型游戏,只是CPU不能差,Intel I7.3G的主频…… 其 实现在CPU ...
- Clusterware后台进程
Clusterware由若干进程组成,其中最重要的是CRSD,CSSD,EVMD 在Clusterware安装的最后阶段,会要求在每个节点执行root.sh脚本,这个脚本实际的作用就是在/etc/ ...
- Java程序执行过程
首先,写好Java代码,保存到硬盘中.然后在命令行中输入: javac ClassName.java 此时,这个Java类文件将编译成字节码(.class)文件.如果用Eclipse等IDE开发工具, ...
- Linux/Unix shell sql 之间传递变量
灵活结合Linux/Unix Shell 与SQL 之间的变量传输,极大程度的提高了DBA的工作效率,本文针对Linux/Unix shell sql 之间传递变量给出几个简单的示例以供参考. Lin ...
- IconRes提供免费高质量的Material风格android官方图标库
连接地址: http://www.iconres.com/android/index.php
- Suse linux 11 SP2 nginx 使用笔记
1. 下载源代码 http://nginx.org/en/download.html 2. DAV模块缺省没有编译,要加入编译选项 # ./configure --with-http_dav_mo ...
- Dubbo原理解析-注册中心之Zookeeper协议注册中心
下面我们来看下开源dubbo推荐的业界成熟的zookeeper做为注册中心, zookeeper是hadoop的一个子项目是分布式系统的可靠协调者,他提供了配置维护,名字服务,分布式同步等服务.对于z ...
- USB鼠标过一段时间后失灵问题的修复
现象: USB鼠标计算机锁屏一段时间后,不能动了,拔下来重新插上后,又恢复正常了. 原因: 这是系统默认USB电源管理造成的.一段时间不用后,自动关闭了USB电源. 解决方法: 1.进入设备管理器 在 ...
- 【LeetCode 235】Lowest Common Ancestor of a Binary Search Tree
Given a binary search tree (BST), find the lowest common ancestor (LCA) of two given nodes in the BS ...