JavaScript优化以及前段开发小技巧


Posted in Javascript onFebruary 02, 2017

一、网络优化

YSlow有23条规则,中文可以参考这里。这几十条规则最主要是在做消除或减少不必要的网络延迟,将需要传输的数据压缩至最少。

1)合并压缩CSS、JavaScript、图片,静态资源CDN缓存

通过构建工具Gulp,可以在开发的时候就将合并压缩的事情一起做掉。

之所以要做合并压缩是因为:HTTP 1.x不允许一个连接上的多个响应数据交错到达(多路复用),因而一个响应必须完全返回后,下一个响应才会开始传输。

也就是说即使客户端同时发送了两个请求,而且CSS资源先准备就绪,服务器也会先发送HTML响应,然后再交付CSS。

JavaScript优化以及前段开发小技巧

使用CDN是为了让用户访问的时候能用最近的资源,减少来回传输时间。

HTTP2.0改进了HTTP1.x很多方面。

2)CSS放顶部,JavaScript放底部

CSS可以并行下载,而JavaScript加载之后会造成阻塞。

但凡事还是会有例外,如果把行内脚本放在样式表之后,会明显地延迟资源的下载(结果是样式表下载完成并且行内脚本执行完毕时,后续资源才能开始下载)。

这是因为行内脚本可能含有依赖于样式表中样式的代码,比如document.getElementsByClassName()。

<head>
 <link rel="stylesheet" href="css/all-normal.css" type="text/css" />
</head>
<body>
 <div id="content"></div>
 <script>
  var content = '';
  for(i=1; i<1000000; i++)
    content += '写入页面';
  document.getElementById('content').innerHTML = content;
 </script>
 <img src="images/ui.png" />
</body>

下面通过Chrome的工具查看下:

JavaScript优化以及前段开发小技巧

3)优化DNS解析,减少重定向

在做一个“女神评选活动”的时候,需要在微信中访问能够获取用户的openid,微信获取用户基本信息是需要经过几个步骤的:

先获取code,再通过code获取openid,最后再跳转访问静态页面。

由于公司将业务分成了多个小组,所以短短的三步其实需要三个小组配合,需要重定向多个域名。

下图是未优化前的瀑布图,但不是最坏情况,有时候访问到静态页面需要经过10多秒,完全不能接受,下图中会跳转4个域名:

JavaScript优化以及前段开发小技巧

后面不跳index那个域名,直接跳转到微信操作域名,减少一个域名的跳转,各小组代码再做优化,但效果还是不理想,仅仅快了几秒。

最后发现其实是在与微信的服务器做交互的时候,DNS解析花了太多时间!不得已,在服务器的host中添加一条记录,直接通过IP指向。

下图是最终优化结果,虽然达不到秒开,但至少可以接受了:

JavaScript优化以及前段开发小技巧

二、JavaScript优化

1)图片预加载

在做一个“秋名山活动”的时候,使用了图片预加载。这个活动中有120多张图片。

流程很简单,就是答题,最后给评论结果,再分享出去。

JavaScript优化以及前段开发小技巧 

JavaScript优化以及前段开发小技巧 

JavaScript优化以及前段开发小技巧

如果一下子加载那么多图片,一定是愚蠢的想法,最后决定,在页面载入的时候先加载一些通用图片。

在答题的时候当前页面,预先加载后面页面中的图片,防止访问页面的时候直接不展示图片,图片也做了适当的合并。

将网站地址放在gtmetrix.com测试,下面是最终的瀑布图,可以发现图片都在其他静态资源的后面,这样能尽早的展现页面给用户:

JavaScript优化以及前段开发小技巧

优化还远远没有结束,在Chrome中分别模拟了good 2G、good 3G以及4G后,有结果的情况并不理想。

good 2G:

JavaScript优化以及前段开发小技巧

good 3G:

JavaScript优化以及前段开发小技巧

4G:

JavaScript优化以及前段开发小技巧

还有很大的优化空间可以做,关于这个预加载的原理,可以参考《图片预加载与懒加载》

 2)减少分支

在写业务逻辑的时候,经常会用到if else,switch之类的逻辑判断,如果每次都做这么多判断,很容易影响性能。

所以可以通过多种方式来避免过多的判断。

1. 惰性模式

这是在看《JavaScript设计模式》的时候看到的。

减少每次代码执行时的重复性分支判断,通过对对象重定义来屏蔽原对象中的分支判断。

惰性模式分为两种:第一种文件加载后立即执行对象方法来重定义,第二种是当第一次使用方法对象时来重定义。

公司有个页面要提供给第三方APP,但是最终发现第三方APP不能使用localStorage缓存,最终只得做兼容的方式。

但为了避免每次引用方法的时候都做判断,就使用加载后立即重定义:

var getFn = function() {
 if (sore.enabled)
  return sore.get;
 return cookie.get;
}();
var setFn = function() {
 if (sore.enabled)
  return sore.set;
 return cookie.set;
}();

2. 建立映射关系

页面中经常需要弹出框提示,后面就自己做了一个,但弹出框会有很多款式。

如果用简单工厂模式创建的话,免不了switch分支判断,后面就直接用赋不同的key,还能缓存起来,只初始化一次。

JavaScript优化以及前段开发小技巧JavaScript优化以及前段开发小技巧

/**
 * 弹出框单例模式
 */
var factories = {};
var DialogFactory = function(type, options) {
 if (factories[type])
  return factories[type];
 return factories[type] = new iDialog(options);
};
/**
 * 提示框
 */
var Alert = function(content, options) {
 var d = DialogFactory('alert', options);
 //其他逻辑省略
 return d;
};
/**
 * 确认框
 */
var Confirm = function(content, options) {
 var d = DialogFactory('confirm', options);
 //其他逻辑省略
 return d;
};

3)第三方代码异步加载

第三方代码,例如百度统计、微信SDK等,这些完全可以在将业务资源加载完后再添加。

/**
 * 百度统计设置
 */
util.baidu = function(key) {
 global._hmt = global._hmt || [];
 (function() {
  var hm = document.createElement("script");
  hm.src = "//hm.baidu.com/hm.js?" + key;
  var s = document.getElementsByTagName("script")[0];
  s.parentNode.insertBefore(hm, s);
 })();
};

4)cookie与localStorage缓存

有了缓存后,就能减少与服务器的通信,在本地操作。

公司有个查违章的业务,在本地添加好车辆后,再次进入页面的时候就需要能直接选择事先添加好的车辆。

最理想的方式就是添加好后,就在本地缓存起来,下次进入直接调取缓存。

  JavaScript优化以及前段开发小技巧

我会优先使用localStorage,下面的表格就是对比:

cookie

localStorage

数据生命周期 

可设置失效时间除非被清除,否则永久保存

数据大

大约4KB 大约5M

与服务器通信

每次都会携带在HTTP头中,如果使用cookie保存过多数据会带来性能问题 
不参与和服务器的通信

本地存储,之前的历史大概如下图所示:

JavaScript优化以及前段开发小技巧

localStorage在浏览器兼容方面,IE8居然也支持了。

JavaScript优化以及前段开发小技巧

5)事件委托

使用事件委托技术能让你避免对特定的每个节点添加事件监听器。

事件监听器是被添加到它们的父元素上,通过事件冒泡,触发执行。

在开发的时候,经常会出现动态添加元素的情况。

如果每次都重新绑定一次事件,那会有很多多余操作,而绑定在此元素的父级,就只需绑定一次即可。

document.getElementById('ul').onclick = function(e) {
 var e = e || window.event,
  tar = e.target || e.srcElement;
 if (tar.nodeName.toLowerCase() == 'li') {
  tar.style.background = 'black';
 }
}
6)节流与去抖动

节流(throttle):预先设定一个执行周期,当调用动作的时刻大于等于执行周期则执行该动作,然后进入下一个新周期。

例如mousemove 事件、window对象的resize和scroll事件。

去抖动(debounce):当调用动作n毫秒后,才会执行该动作,若在这n毫秒内又调用此动作则将重新计算执行时间。

例如文本输入keydown 事件,keyup 事件,做autocomplete等。

节流与去抖动最大的不同的地方就是在计算最后执行时间的方式上。著名的开源工具库underscore中有内置了两个方法。

在做公司内部的一个系统的时候,需要方希望在左右滚动表格的时候,能将第一列固定在最左边,方便查看。

为了让操作能更流畅,我再这里用了节流,有些浏览器会出现卡顿,就得需要增加周期时间。

JavaScript优化以及前段开发小技巧

三、小技巧

1)在手机中打印变量

在移动页面的时候经常需要调试字段,又不能用console.log,每次alert的话,碰到对象就看不到内容了。

只能自己写个小方法来打印出来,JSON.stringify,通过这个方法能够方便的实现功能。

var print = function(obj, space) {
 space = space || 4;
 var html = JSON.stringify(obj, null, space);
 html = html.replace(/\n/g, '<br>').replace(/\s/g, ' ');
 var pre = document.createElement('pre');
 var div = document.createElement('code');
 pre.style.cssText = 'border:1px solid #000;padding:10px;background:#FFF;margin-bottom:20px;';
 div.innerHTML = html;
 pre.appendChild(div);
 var body = document.querySelector('body');
 body.insertBefore(pre, body.children[0]);
};

print({a:1, b:'demo', c:{text:'content'}});

JavaScript优化以及前段开发小技巧

 2)chrome插件JSON-handle

服务器返回的很多都是JSON格式的数据,通常写好后给你个接口,顺便给你几个demo参数。

在浏览器中打开后,就是一串字符串,但要给人看的话,就得格式化一下了,这个插件就是用来让人看的。

JavaScript优化以及前段开发小技巧

Javascript 相关文章推荐
js中将HTMLCollection/NodeList/伪数组转换成数组的代码
Jul 31 Javascript
javascript上传图片前预览图片兼容大多数浏览器
Oct 25 Javascript
禁止IE用右键的JS代码
Dec 30 Javascript
jQuery CSS()方法改变现有的CSS样式
Aug 20 Javascript
JavaScript中用字面量创建对象介绍
Dec 31 Javascript
jQuery插件Tooltipster实现漂亮的工具提示
Apr 12 Javascript
轻量级javascript 框架Backbone使用指南
Jul 24 Javascript
JavaScript实现算术平方根算法-代码超简单
Sep 11 Javascript
学JavaScript七大注意事项【必看】
May 04 Javascript
浅谈String.valueOf()方法的使用
Jun 06 Javascript
Javascript中的arguments对象
Jun 20 Javascript
封装一下vue中的axios示例代码详解
Feb 16 Javascript
JavaScript字符集编码与解码详谈
Feb 02 #Javascript
JS实现购物车特效
Feb 02 #Javascript
jQuery实现复选框的全选和反选
Feb 02 #Javascript
jQuery制作图片旋转效果
Feb 02 #Javascript
浅谈javascript中的 “ &amp;&amp; ” 和 “ || ”
Feb 02 #Javascript
Javascript中的 “&amp;” 和 “|” 详解
Feb 02 #Javascript
javascript实现简易计算器
Feb 01 #Javascript
You might like
redis+php实现微博(一)注册与登录功能详解
2019/09/23 PHP
IE Firefox 使用自定义标签的区别
2009/10/15 Javascript
JQuery下的Live方法和$.browser方法使用代码
2010/06/02 Javascript
解读JavaScript代码 var ie = !-[1,] 最短的IE判定代码
2011/05/28 Javascript
js网页实时倒计时精确到秒级
2014/02/10 Javascript
raphael.js绘制中国地图 地图绘制方法
2014/02/12 Javascript
使用JQuery库提供的扩展功能实现自定义方法
2014/09/09 Javascript
js控制鼠标事件移动及移出效果显示
2014/10/19 Javascript
Bootstrap每天必学之折叠(Collapse)插件
2016/04/25 Javascript
Augularjs-起步详解
2016/07/08 Javascript
jQuery实现右键菜单、遮罩等效果代码
2016/09/27 Javascript
JavaScript之排序函数_动力节点Java学院整理
2017/06/30 Javascript
Vue的Class与Style绑定的方法
2017/09/01 Javascript
jQuery实现倒计时功能 jQuery实现计时器功能
2017/09/19 jQuery
微信小程序实现的一键复制功能示例
2019/04/24 Javascript
如何利用node.js开发一个生成逐帧动画的小工具
2019/12/01 Javascript
在Vue中实现随hash改变响应菜单高亮
2020/03/09 Javascript
Python中的深拷贝和浅拷贝详解
2015/06/03 Python
numpy使用技巧之数组过滤实例代码
2018/02/03 Python
Anaconda入门使用总结
2018/04/05 Python
详解windows python3.7安装numpy问题的解决方法
2018/08/13 Python
Python实现E-Mail收集插件实例教程
2019/02/06 Python
Python进度条的制作代码实例
2019/08/31 Python
咖啡为什么会有酸味?你喝到的咖啡為什麼是酸的?
2021/03/17 冲泡冲煮
Fossil美国官网:化石手表、手袋、首饰及配饰
2019/02/17 全球购物
铭宣海淘转运:美国、日本、英国转运等全球转运公司
2019/09/10 全球购物
初中政治教学反思
2014/01/17 职场文书
2014三八妇女节活动总结范文四篇
2014/03/09 职场文书
供货协议书范本
2014/04/22 职场文书
我的画教学反思
2014/04/28 职场文书
爱耳日宣传活动总结
2014/07/05 职场文书
小学生感恩老师演讲稿
2014/08/28 职场文书
户籍证明模板
2014/09/28 职场文书
Javascript之datagrid查询详解
2021/09/15 Javascript
Java数据结构之堆(优先队列)
2022/05/20 Java/Android
IDEA中sout快捷键无效问题的解决方法
2022/07/23 Java/Android