前端开发最容易犯的13个JavaScript错误

简介:

前端开发最容易犯的13个JavaScript错误,你中标了没

开发者最容易犯的JavaScript错误,总结出13个。这些当中可能少不了你犯的错误^_^。我们描述了这些陋习,并列出来解决办法,希望对开发者有帮助。

1.for.. 数组迭代的用法 Usage of for..in to iterate Arrays 举例: var myArray = [ a, b, c ]; var totalElements = myArray.length; for (var i = 0; i totalElements; i++) { console.log(myArray[i]); }

这里主要的问题是语句中的for…不能保证顺序,这意味着你将获得不同的执行结果。此外,如果有人增加一些其他自定义功能的函数Array.prototype,你的循环将重复遍历这些函数,就像原数组项。

解决办法:一直使用规则的for循环来遍历数组。 var myArray = [ a, b, c ]; for (var i=0; imyArray.length; i++) { console.log(myArray[i]); }

2. 数组维度 Array dimensions 举例 var myArray = new Array(10);

这里有两个不同的问题。首先,开发者尝试创建一个包含10项的数组,这将创建10个空槽的阵列。然而,如果你试图得到一数组项,你将得到未定义的结果。换句话说,效果就像你没有保存内存空间。没有真正的好原因来预定义数组长度。

第二个问题是开发者使用数组构成器来创建数组,技术上是正确的,然而会比文字符号(literal notation)慢

解决办法:使用文字符号来初始化数组,不要预定义数组长度。 var myArray = [];

3. 未定义属性 Undefined properties 举例: var myObject = { someProperty: value, someOtherProperty: undefined }

未定义属性,将在对象中创建元素( 键’someOtherProperty’ 和 值 ‘undefined’.)。如果你遍历数组,检测已存在的元素,那么下面的语句将都返回未定义/undefined typeof myObject['someOtherProperty'] // undefined typeof myObject['unknownProperty'] // undefined

解决办法: 如果你想明确声明对象中的未初始化的属性,标记它们为Null(空)。 var myObject = { someProperty: value, someOtherProperty: null }

4. 闭包的滥用 Misuse of Closures 举例: function(a, b, c) { var d = 10; var element = document.getElementById(‘myID’); element.onclick = (function(a, b, c, d) { return function() { alert (a + b + c + d); } })(a, b, c, d); }

这里开发者使用两个函数来传递参数a、b、c到onclick handler。双函数根本不需要,徒增代码的复杂性。

变量abc已经在局部函数中被定义,因为他们已经在主函数中作为参数被声明。局部函数中的任何函数都可创建主函数中定义的所有变量的闭包。因此不需要再次传递它们。

看看这里 JavaScript Closures FAQ 了解更多。

解决办法:使用闭环来简化你的代码。 function (a, b, c) { var d = 10; var element = document.getElementById(‘myID’); element.onclick = function() { //a, b, and c come from the outer function arguments. //d come from the outer function variable declarations. //and all of them are in my closure alert (a + b + c + d); }; }

5. 循环中的闭包 Closures in loops 举例: var elements = document.getElementByTagName(‘div’); for (var i = 0; ielements.length; i++) { elements[i].onclick = function() { alert(Div number + i); } }

在这里例子里面,当用户点击不同的divs时,我们想触发一个动作(显示Div number 1, Div number 2… 等) 。然而,如果你在页面有10个divs,他们全部都会显示 Div number 10。

问题是当我们使用局部函数创建一个闭包时,函数中的代码可以访问变量i。关键是函数内部i和函数外部i涉及同样的变量。当我们的循环结束,i指向了值10,所以局部函数中的i的值将是10。

解决办法:使用第二函数来传递正确的值。 var elements = document.getElementsByTagName(‘div’); for (var i = 0; ielements.length; i++) { elements[i].onclick = (function(idx) { //Outer function return function() { //Inner function alert(Div number + idx); } })(i); }

6. DOM对象的内测泄漏 Memory leaks with DOM objects 举例: function attachEvents() { var element = document.getElementById(‘myID’); element.onclick = function() { alert(Element clicked); } }; attachEvents();

该代码创建了一个引用循环。变量元素包含函数的引用(归于onclick属性)。同时,函数保持一个DOM元素的引用(提示函数内部可以访问元素, 因为闭包。)。所以JavaScript垃圾收集器不能清除元素或是函数,因为他们被相互引用。大部分的JavaScript引擎对于清除循环应用都不够 聪明。

解决办法:避免那些闭包,或者不去做函数内的循环引用。 function attachEvents() { var element = document.getElementById(‘myID’); element.onclick = function() { //Remove element, so function can be collected by GC delete element; alert(Element clicked); } }; attachEvents();

7. 区别整数数字和浮点数字 Differentiate float numbers from integer numbers 举例: var myNumber = 3.5; var myResult = 3.5 + 1.0; //We use .0 to keep the result as float

在JavaScript中,浮点与整数间没有区别。事实上,JavaScript中的每个数字都表示使用双精度64位格式IEEE 754。简单理解,所有数字都是浮点。

解决办法:不要使用小数(decimals),转换数字(numbers)到浮点(floats)。 var myNumber = 3.5; var myResult = 3.5 + 1; //Result is 4.5, as expected

8. with()作为快捷方式的用法 Usage of with() as a shortcut 举例: team.attackers.myWarrior = { attack: 1, speed: 3, magic: 5}; with (team.attackers.myWarrior){ console.log ( Your warrior power is + (attack * speed)); }

讨论with()之前,要明白JavaScript contexts 如何工作的。每个函数都有一个执行 context(语句),简单来说,包括函数可以访问的所有的变量。因此 context 包含 arguments 和定义变量。

with() 真正是做什么?是插入对象到 context 链,它在当前 context 和父级 context间植入。就像你看到的with()的快捷方式会非常慢。

解决办法:不要使用with() for shortcuts,仅for context injection,如果你确实需要时。 team.attackers.myWarrior = { attack: 1, speed: 3, magic: 5}; var sc = team.attackers.myWarrior; console.log(Your warrior power is + (sc.attack * sc.speed));

9.setTimeout/setInterval 字符串的用法 Usage of strings with setTimeout/setInterval 举例: function log1() { console.log(document.location); } function log2(arg) { console.log(arg); } var myValue = test; setTimeout(log1(), 100); setTimeout(log2( + myValue + ), 200);

setTimeout() 和 setInterval() 可被或一个函数或一个字符串作为首个参数。如果你传递一个字符串,引擎将创建一个新函数(使用函数构造器),这在一些浏览器中会非常慢。相反,传递函数本身作为首个参数,更快、更强大、更干净。

解决办法: 一定不要使用 strings for setTimeout() 或 setInterval()。 function log1() { console.log(document.location); } function log2(arg) { console.log(arg); } var myValue = test; setTimeout(log1, 100); //Reference to a function setTimeout(function(){ //Get arg value using closures log2(arg); }, 200);

10.setInterval() 的用法 Usage of setInterval() for heavy functions 举例: function domOperations() { //Heavy DOM operations, takes about 300ms } setInterval(domOperations, 200);

setInterval() 将一函数列入计划被执行,仅是在没有另外一个执行在主执行队列中等待。JavaScript 引擎只增加下一个执行到队列如果没有另外一个执行已在队列。这可能导致跳过执行或者运行2个不同的执行,没有在它们之间等待200ms的情况下。

一定要搞清,setInterval() 没有考虑进多长时间domOperations() 来完成任务。

解决办法:避免 setInterval(),使用 setTimeout() function domOperations() { //Heavy DOM operations, takes about 300ms //After all the job is done, set another timeout for 200 ms setTimeout(domOperations, 200); } setTimeout(domOperations, 200);

11. this的滥用 Misuse of ‘this’ 这个常用错误,没有例子,因为非常难创建来演示。this的值在JavaScript中与其他语言有很大的不同。

函数中的this值被定义是在当函数被调用时,而非声明的时间,这一点非常重要。下面的案例中,函数内this有不同的含义。 * Regular function: myFunction(‘arg1’);

this points to the global object, wich is window for all browers.

  • Method: someObject.myFunction(‘arg1’);

this points to object before the dot, someObject in this case.

  • Constructor: var something = new myFunction(‘arg1’);

this points to an empty Object.

  • Using call()/apply(): myFunction.call(someObject, ‘arg1’);

this points to the object passed as first argument.

12. eval()访问动态属性的用法 Usage of eval() to access dynamic properties 举例: var myObject = { p1: 1, p2: 2, p3: 3}; var i = 2; var myResult = eval(‘myObject.p’+i);

主要问题在于使用eval() 开始一个新的执行语句,会非常的慢。

解决办法:使用方括号表示法(square bracket notation)代替 eval()。 var myObject = { p1: 1, p2: 2, p3: 3}; var i = 2; var myResult = myObject[p+i];

13. 未定义(undefined)作为变量的用法 Usage of undefined as a variable 举例: if ( myVar === undefined ) { //Do something }

在上面的例子中,未定义实际上是一变量。所有的JavaScript引擎会创建初始化的变量window.undefined 给未定义作为值。然而注意的是变量不仅是可读,任何其他的代码可以刚改它的值。很奇怪能找到window.undefined 有来自未定义的不同的值的场景,但是为什么冒险呢?

解决办法:检查未定义时,使用typeof。 if ( typeof myVar === undefined ) { //Do something } h

对于前端学习有不懂的,或者遇到学习瓶颈,不知道学习方法,小编整理了最新的JavaScript、jQuery、bootstrap、angularJS、react、nodejs等企业级框架教程,还有PDF文档资料都上传到网盘了,来帮助大家一起成长。 地址:https://www.dongnaoedu.com/web/web.html

原文发布时间:2018年01月17日

作者:李红欧巴

本文来源:前端乱炖  如需转载请联系原作者

目录
相关文章
|
6天前
|
JavaScript 前端开发 程序员
前端原生Js批量修改页面元素属性的2个方法
原生 Js 的 getElementsByClassName 和 querySelectorAll 都能获取批量的页面元素,但是它们之间有些细微的差别,稍不注意,就很容易弄错!
|
1月前
|
JavaScript 前端开发 程序员
前端学习笔记——node.js
前端学习笔记——node.js
43 0
|
11天前
|
资源调度 前端开发 JavaScript
vite3+vue3 实现前端部署加密混淆 javascript-obfuscator
【11月更文挑战第10天】本文介绍了在 Vite 3 + Vue 3 项目中使用 `javascript-obfuscator` 实现前端代码加密混淆的详细步骤,包括安装依赖、创建混淆脚本、修改 `package.json` 脚本命令、构建项目并执行混淆,以及在 HTML 文件中引用混淆后的文件。通过这些步骤,可以有效提高代码的安全性。
|
19天前
|
设计模式 前端开发 JavaScript
揭秘!前端大牛们如何巧妙利用JavaScript,打造智能交互体验!
【10月更文挑战第30天】前端开发领域充满了无限可能与创意,JavaScript作为核心语言,凭借强大的功能和灵活性,成为打造智能交互体验的重要工具。本文介绍前端大牛如何利用JavaScript实现平滑滚动、复杂动画、实时数据更新和智能表单验证等效果,展示了JavaScript的多样性和强大能力。
34 4
|
17天前
|
机器学习/深度学习 自然语言处理 前端开发
前端神经网络入门:Brain.js - 详细介绍和对比不同的实现 - CNN、RNN、DNN、FFNN -无需准备环境打开浏览器即可测试运行-支持WebGPU加速
本文介绍了如何使用 JavaScript 神经网络库 **Brain.js** 实现不同类型的神经网络,包括前馈神经网络(FFNN)、深度神经网络(DNN)和循环神经网络(RNN)。通过简单的示例和代码,帮助前端开发者快速入门并理解神经网络的基本概念。文章还对比了各类神经网络的特点和适用场景,并简要介绍了卷积神经网络(CNN)的替代方案。
|
17天前
|
移动开发 前端开发 JavaScript
前端实训,刚入门,我用原生技术(H5、C3、JS、JQ)手写【网易游戏】页面特效
于辰在大学期间带领团队参考网易游戏官网的部分游戏页面,开发了一系列前端实训作品。项目包括首页、2021校园招聘页面和明日之后游戏页面,涉及多种特效实现,如动态图片切换和人物聚合效果。作品源码已上传至CSDN,视频效果可在CSDN预览。
27 0
前端实训,刚入门,我用原生技术(H5、C3、JS、JQ)手写【网易游戏】页面特效
|
22天前
|
JavaScript 前端开发 开发者
前端框架对比:Vue.js与Angular的优劣分析与选择建议
【10月更文挑战第27天】在前端开发领域,Vue.js和Angular是两个备受瞩目的框架。本文对比了两者的优劣,Vue.js以轻量级和易上手著称,适合快速开发小型到中型项目;Angular则由Google支持,功能全面,适合大型企业级应用。选择时需考虑项目需求、团队熟悉度和长期维护等因素。
32 1
|
23天前
|
JavaScript 前端开发 API
前端框架对比:Vue.js与Angular的优劣分析与选择建议
【10月更文挑战第26天】前端技术的飞速发展让开发者在构建用户界面时有了更多选择。本文对比了Vue.js和Angular两大框架,介绍了它们的特点和优劣,并给出了在实际项目中如何选择的建议。Vue.js轻量级、易上手,适合小型项目;Angular结构化、功能强大,适合大型项目。
21 1
|
26天前
|
前端开发 JavaScript UED
"前端小技巧大揭秘:JS如何将后台时间戳秒变亲切小时前、分钟前,让用户秒懂,提升互动体验!"
【10月更文挑战第23天】在Web开发中,将后台返回的时间戳转换为“小时前”、“分钟前”、“刚刚”等友好的时间描述是常见需求。本文介绍如何用JavaScript实现这一功能,通过计算当前时间和时间戳的差值,返回相应的描述,提升用户体验。
28 1
|
1月前
|
前端开发 JavaScript 安全
JavaScript前端开发技术
JavaScript(简称JS)是一种广泛使用的脚本语言,特别在前端开发领域,它几乎成为了网页开发的标配。从简单的表单验证到复杂的单页应用(SPA),JavaScript都扮演着不可或缺的角色。
21 3
下一篇
无影云桌面