`
king_jw
  • 浏览: 4789 次
  • 性别: Icon_minigender_1
  • 来自: 江西
最近访客 更多访客>>
社区版块
存档分类
最新评论

Js 调错技巧

    博客分类:
  • JS
阅读更多

任何一个编程者都少不了要去调试代码,不管你是高手还是菜鸟,调试程序都是一项必不可少的工作。一般来说调试程序是在编写代码之后或测试期修改Bug 时进行的,往往在调试代码期间更加能够体现出编程者的水平高低以及分析问题的准确度。不少初学者在寻找错误原因时,总是不得要领,花费了大量时间却无法解决一些最终证明是相当简单的Bug。

 

在长期解答 zTree 相关问题时,也的确发现很多的问题其实不算什么问题,仅仅是编程者不会调试造成的,通过自己日常工作观察,这里面有态度问题也有思考问题的方式方法,故总结一下自己经验,以供大家参考(尤其是菜鸟), 对于高手来讲,如果你愿意看完这篇文章,也非常欢迎拍砖和提供建议,我相信这样可以不断让这篇文章充实并完善起来,肯定也可以帮助到更多的朋友。

 

试想一下:出现了某个bug,有人用几分钟就搞定了,有人用了半天或者一天都找不到原因所在。你愿意当前者还是后者呢?想当前者的就请好好看完本篇文章吧。

 

本篇文章主要是帮助大家在调试代码时,如何建立解决问题的思路。所以并不会专门介绍不同浏览器的具体调试方法。(在网上关于 chrome、firefox、ie 等浏览器的具体调试方法已经有很多了,这里就不一一描述啦。)另外,俗话说的好——“js、html 和 css 是一家”,因此在讲述中也会稍带有html 和 css 的内容。

 

一、遇到错误怎么办?

辛辛苦苦写了一大段代码却无法运行,怎么办?……前两天还正常的代码怎么今天报错了,怎么办?……同样的代码在这个页面正常怎么到了另一页面就出问题了,怎么办?……这些情况你都遇到过吗?是否被错误搞得焦头烂额、心情烦躁?(“不管路途有多么遥远,有多少艰难险阻,都不可能阻挡我前进的步伐!”——唐师傅)

 

1、调整心态

出现问题很正常,其实不出问题反倒不太正常。所以当发现问题的时候我们需要冷静。不要在乎身后站着的是谁,不要在乎距离上线时间还有几个小时,让自己平静下来,深呼一口气——来吧,不就是个小问题嘛。所有的Bug都在我的掌控之中!

禁忌:出了问题就找人帮忙,这样会导致经常打扰别人,而且也不利于自己的技能提升。只有当自己经过一番努力后,的确找不到解决办法时再去寻求帮助。而且提出问题时,也要尽量将问题描述清楚。

 

2、寻找特征

解决错误的前提是要发现错误,发现错误的前提嘛…当然是要去“寻找”喽!对于简单的报错信息完全可以利用调试工具的提示:xxxx行出现什么什么错误。如果这则信息100%有效,那么你就不需要再看这篇文章了。

我们遇见的错误表象一般分为以下几种

 

  • 直接报告 js 语法错误

 

这种一般最容易解决,不需要我来废话了……

 

  • js 报错,但报错地点不是出问题的根源

 

这种情况大部分可以解决,但有时候完全不知道是从哪里引用过来的,这种时候会相当的头疼。

 

  • js 无报错,但功能无效

 

这种情况更糟糕,完全没有头绪了……那我告诉你,往往这种时候最终解决错误的方法更容易。

根据特征,往往能够快速定位错误的大概位置,便于进一步查找问题。

 

3、怀疑一切

当有人告诉你代码有问题时,我们的第一反应经常是:“不可能!”,“你是不是看错了?”,“我刚才运行还好好地呢!”如果你有以上想法,那么需要注意喽!这些想法很危险,如果想解决问题,那么我们就要去怀疑任何有可能的事情,要以特征为主,不要主观断定哪些地方肯定不会出错。

 

二、如何让错误现出原形?

前面主要是为了让我们能有一个良好的心态来处理问题,情绪保持冷静可以让我们的思维更加敏捷,抓住特征可以让我们更快的找到线索,怀疑一切可以让我们有更多的思路去发现错误。(“不把妖孽打得显出原形,就别想翻过这座山!”——猴哥)

 

1、化繁为简

临床表现:莫名其妙的报错,不方便调试,单纯看代码无法解释出错原因。

主要病因:js脚本冲突、Css冲突、DOM的ID冲突、DOM标签缺失等多种由于冲突产生的bug根源

排查方法:熟练使用 Delete / Backspace 键,对代码不断进行区域删除,直到症状消失,最后一次删除的区域很可能就是导致错误的根源。进一步查找根源可以在目标区域使用更小单位的区域定义反复使用此方法。

注意:

 

  • 对于 js 代码建议分别以类库  功能  行 为单位进行删除测试
  • 对于 HTML 代码建议按照页面结构先删除最内部的 tag,由内及外,这样可以迅速发现由于 tag 缺失造成的错误,同时也能保证每次删除的都是整段的代码,避免由于删除产生新的错误。
  • 对于 css 代码建议按照css文件  定义的class系列  行 为单位批量进行删除测试

 

忌:对于确认与相关功能有关的部分不要随便删除。

副作用:使用此方法也可以很好的确认错误原因是js造成的还是css造成的。

 

2、顺藤摸瓜

临床表现:错误信息较准确,能够按照提示的错误逐层跟踪,使用“化繁为简”能够基本定位的错误类型

主要病因:基本语法错误、逻辑错误、不严谨(最常见的有:数组下标越界,null空指针导致的对象找不到,undefined未初始化,NaN数字计算错误等)

排查方法:利用浏览器的调试工具(F12是个很好的功能键):跟踪代码;利用console.log 输出监控对象(IE无效);利用 alert监控(最无奈的方法,用于极端情况);把调试代码加载报错命令行的前面,一般都会有奇效!

注意:

 

  • 对于无js 报错的情况,直接调试无效功能的代码即可。有时候会发现之所以功能无效的原因居然是没有调用该代码!(是不是很可笑?我承认我犯过这种错误。)
  • 逻辑错误往往不太容易想清楚,这时候可以适当结合“化繁为简”的思路进行调试。
  • 当你使用的是类似于 jQuery 这种js库时,如果报错信息处于js库内,首先要更换为未压缩的js代码进行调试,然后分析是自己的哪段代码会调用相关功能。

 

忌:钻牛角尖!当按照此方法仍无法找到错误根源时,说明这并不是一个准确的错误信息,肯定有其他潜在的因素在产生错误。立刻更改切入点,不要在一个地方长时间浪费时间。(这种时候,可以参考下一个方法:“反复对照”)

 

3、反复对照

临床表现: 应用普通方法很难定位错误,前两种方法怎么用都还是找不到头绪。

主要病因: 逻辑复杂、功能互相绑定难以剥离、页面对象内容复杂、有的页面正常有的页面不正常、兼容问题等

排查方法: 对付这种复杂的麻烦,排查方法也会很繁琐,但并不是什么特别高深的技术,只是需要多做一些体力活儿而已。

 

  • 方法一:检查可疑的代码,细化功能点,每次只修改一个地方,修改一次就测试一次,直到发现导致错误的关键代码。
  • 方法二:制作最简单的Demo 只实现需要的功能,当功能正常后,与出错的正式代码进行比较(比较中可以适当使用方法一)
  • 方法三:以正常功能(或出错)的代码为原型,修改一个地方就生成一个测试的备份,每个测试的案例都只有一处与原始代码不同,编上号码,在特殊环境下逐一测试,检查导致错误的根源(我曾经用这个方法解决了韩文系统下IE8加载公司内部flash 不能正常发声的bug)

 

注意:

 

  • 使用此法一定要有耐心
  • 此方法技术含量低,任何人都可以快速掌握,解决某些疑难杂症基本上可以说是药到病除!

 

忌: 急躁、马虎

 

4、积累经验

临床表现:部分浏览器报错、事件响应异常、js操作DOM无效、PC 和 触屏系统功能不一致等

主要病因:各种兼容问题

排查方法:对于某些有明显特征的报错要熟记于胸,看到这些情况能立刻想到应该是哪些原因导致的。(最明显的例子:Json对象多了一个逗号的情况,只有ie报错)

注意:

 

  • 有个很基础的问题,但有很多初学者出错——没有搞清楚页面 html、css、js 的加载顺序,导致js 操作失败。(去Google 或 Baidu 搜索:“html css js 加载顺序”)
  • 日常工作要细心,勤观察。认真对待每一次查找错误的工作,对于部分非常特殊的情况可以记下来。
  • 熟练利用 Google、Baidu 等搜索引擎,有时候自己第一次遇到的情况,别人早都知道如何解决了。

 

忌:粗心大意、不求甚解

 

5、细节决定成败

以上四种查找错误的方法全部都依赖于一个核心——细节!细节往往比你的技术水平更重要。说一句夸张点儿的话,能有多少那么高深的技术等着你去做?好好把自己手上的工作认真完成吧!但请记住了,当你重视细节以后,你距离去做高深技术的机会应该也就不远了。

 

三、如何修正错误?

老程序员们应该已经深有体会,改Bug 最头疼的还是找错,一旦找到错误后,真正解决问题可能真的是只有几分钟。下面针对一些常见的错误原因与修改思路总结一下:

 

1、基本语法、语言基础

逗号、分号、双引号、单引号 以及 各种括号 估计都曾导致过你的代码错误吧?这些东西记牢,必须知道什么时候应该使用什么,不要为了简化代码而精简这些符号。

例如:if / for 等语句后面的 { } 最好还是带上吧。

补充,对于js中的Number 数字的范围希望大家有一定的了解,因为这个范围肯定和后台语言中的Long不一样的。(已经有不止一个朋友跟我说 zTree 会自动修改节点的id,当我看到案例后,原来是数字溢出了!)

 

2、条件严谨

对 Array 或者操作对象属性时,尽可能让条件判断语句写的完整、全面一些。

例如:判断 a.abc 的时候,最好别忘了判断 a 是否存在;或者操作Array时先判断一下Array 是否存在,要操作的下标是否越界等。

 

3、注意兼容(css & js)

警惕部分浏览器不一致或者出错的情况,很多情况都是兼容造成的。如果自己经验不够,直接去Google 或者 Baidu。慢慢的自己经常接触到的一些兼容问题就会牢记下来了。

补充,有时候要注意页头 W3C的定义,曾经有朋友问我 zTree 异常的问题,最终发现是 W3C只写了一半。

 

4、逻辑陷阱

条件过于复杂;循环、判断反复嵌套都是容易导致逻辑陷阱的因素。遇到这种情况,如果自己实在无法解决那么请个身边的高手来吧,让他帮你讲解一下。我相信,除了那些有点儿自闭的人来说,别人都会愿意帮助你解决问题的(前提是你自己别太遭人恨了,呵呵)

补充:多去看看有关 重构 的技术书籍!会让你提高很多的。

 

5、异步加载

其实异步加载出现的问题往往属于逻辑陷阱,但我必须要提出来专门讲述,因为在这上面犯错的人太多了!!!!这里专门详细讲述一遍处理方法:

 

  • 对于异步加载出现了异常,请按照以下流程进行逐一排查:

1)页面是否有报错,是否执行到 ajax 部分的代码?(如果正常请看下一条)

 

2)用浏览器的调试工具监控网络,ajax 加载的url 地址是否正确?(如果正常请看下一条)

 

3)用浏览器的调试工具监控网络,传递给url 的参数是否正确?(如果正常请看下一条)

 

4)用浏览器的调试工具监控网络,从url 返回的数据是否正确?(如果正常请看下一条)

 

5)在 ajax 的success中编写调试代码,调试异步加载后的处理方法是否正确

如果以上几步都正常,那么我可以告诉你异步加载本身是一切正常的,还有错怎么办?继续往下看

 

  • 当异步加载确定正常后,就需要考虑另一个重要问题,也就是我在上一篇文章《这些年我们爱犯的弱智错误(菜鸟必看)》中专门讲述的易犯的错误——异步加载的疏忽

1)千万不要在执行了 ajax 之后立刻去执行应该在异步加载完成之后再运行的代码。因为你执行这段代码的时候,ajax根本没有完成呢。(最明显的现象:运行时时好时坏,但如果我加入了alert 会发现每次都正常了)

这种情况,请将你的代码转移到 ajax的 success 或 error 里面去执行

 

2)当你设置了某些特殊开关时,一定不要忘了在 ajax 的error 里面进行重置,否则很可能因为一次网络异常,就会造成你的页面js 功能失效。这种错误常常是地雷级别的,很难被发现。

 

6、神奇的setTimeout

对于移动设备 或者 某些特殊情况,可以适当考虑使用 setTimeout 来解决问题。

我遇到比较特例的情况:有两个js的事件因为不同的功能在同时对同一个DOM操作时会导致IE8崩溃,显然这是IE的bug,但我无法去要求微软做什么…最终使用 setTimeout 让其中一个功能延迟100-200毫秒再执行,轻松搞定!

 

7、别在一棵树上吊死

做前端的人都很郁闷要适配n多的浏览器,往往会遇到一些自己无法解决的问题(因为是浏览器bug 造成的),遇到这种情况怎么办?一般来说寻找一下有没有 hacker的方法,如果没有,那么就换一种思路,看看是否可以有其他方案来实现类似的功能。如果各种努力都做了…还有人不满意的话,那么告诉他浏览器的bug,让项目管理者来决定到底如何处理吧——这种情况死而无憾了!

分享到:
评论

相关推荐

    js回调函数的使用技巧和认识

    js回调函数js回调函数js回调函数js回调函数js回调函数js回调函数js回调函数...技巧和认识js回调函数的使用技巧和认识js回调函数的使用技巧和认识js回调函数的使用技巧和认识js回调函数的使用技巧和认识js回调函数的使用...

    js技巧 javaScript编码技巧

    以上技巧和知识点在200个js技巧.docx文档中可能有更详尽的解释和实例,建议深入学习,结合实践以提升JavaScript编程技能。同时,不断关注JavaScript的最新发展,如ES规范的更新,以及社区的最佳实践,将有助于你成为...

    JS调式工具

    JS调式工具允许开发者在运行时检查、修改和控制JavaScript代码,这有助于快速定位和解决代码中的错误,提升开发效率。它们通常包括断点设置、步进执行、变量查看、调用堆栈分析等功能。 二、Firefox开发者工具 Fire...

    JS技巧——日常常用JAVASCRIPT脚本

    JavaScript,也被称为JS,是一种广泛应用于网页和网络应用的编程语言,主要负责客户端的动态交互。作为前端开发的重要工具,JavaScript具有轻量级、解释型和面向对象的特点,使得它在网页开发中不可或缺。本文将深入...

    JS技巧大全 javascript

    这个“JS技巧大全”将涵盖一系列实用的JavaScript编程技巧,旨在帮助开发者提升效率,优化代码,并掌握JavaScript的高级特性。 1. **变量声明与作用域** - 使用`let`和`const`替代`var`,以防止变量污染全局作用域...

    js技巧大全,一看就明白

    JavaScript(简称JS)是一种广泛应用于Web开发的轻量级编程语言,它主要负责网页的动态交互,赋予静态HTML页面活力。本篇文章将深入探讨一些重要的JS技巧,旨在帮助初学者和经验丰富的开发者更好地理解和运用这一...

    45个实用的JS技巧

    本文将分享一系列JavaScript的技巧、窍门和最佳实践,这些知识对于所有JavaScript开发者来说,无论是在浏览器/引擎端还是在服务器端JavaScript解释器中都应该知晓。 1. 使用var关键字:在JavaScript中,首次为变量...

    Node.js硬实战 115个核心技巧.pdf

    Node.js是基于Chrome V8引擎的JavaScript运行环境,它以其非阻塞I/O、事件驱动的特性在服务器端编程领域独树一帜,尤其适合构建高性能的网络应用。本书通过115个关键技巧的讲解,全面覆盖了Node.js的基础到高级应用...

    一套基于纯js实现的JavaScript典型应用技巧例子源码

    9. **异步编程**:JavaScript中的异步编程模式有回调函数、Promise、async/await等,它们用于处理耗时操作,避免阻塞主线程。 10. **前端框架预处理**:虽然本套源码未明确提及,但了解如何使用如React、Vue、...

    JavaScript编程技巧

    JavaScript,简称为JS,是一种广泛应用于网页和网络应用的脚本语言,主要负责网页的动态化和交互功能。它基于ECMAScript规范,是Web开发不可或缺的一部分。在本压缩包中,"JavaScript编程技巧"提供了丰富的函数示例...

    搜集的js实践小技巧

    下面,我们将深入探讨一些实用的JavaScript实践小技巧,这些技巧可以帮助开发者提升代码效率、可读性和维护性。 1. **立即执行函数表达式(IIFE)**:在JavaScript中,IIFE是一种常见的封装技术,避免全局变量污染...

    ocx中事件函数,调用js中的回调函数

    总的来说,OCX调用JS中的回调函数是一种典型的客户端编程技巧,它让桌面应用级别的功能得以在Web页面上实现,对于开发复杂交互的网页应用来说非常有用。通过学习和掌握这一技术,开发者能够更好地融合两种语言的优势...

    5分钟就能学会JS的9大技巧_java_技巧_javascript_

    JavaScript,简称JS,是Web开发中的重要脚本语言,它为网页添加动态功能,使得用户交互更为丰富。本文将深入探讨在5分钟内可以掌握的9大JavaScript技巧,旨在帮助初学者快速上手并提升技能。 1. 变量声明与数据类型...

    JavaScript小技巧全集

    本资料集合了众多JavaScript的小技巧,旨在帮助开发者提升工作效率,增强代码质量。 1. 变量与数据类型 - JavaScript中的变量可以通过`var`, `let`或`const`声明。理解它们之间的区别(作用域、可变性)是基础中的...

    JavaScript方法和技巧大全

    异步编程是现代JavaScript的重要特性,通过使用回调函数、Promise或async/await可以实现非阻塞的操作: ```javascript async function fetchUserData(userId) { const response = await fetch(`...

    js技巧 js错误对照表

    JavaScript,简称JS,是Web开发中的重要脚本语言,用于实现客户端的动态效果和交互。在编程过程中,理解和处理JS错误至关重要,因为它们可以帮助我们识别并修复代码中的问题。以下是一些关于JS技巧和错误对照表的...

    JavaScript功能技巧事例

    JavaScript是一种解释型、基于原型的对象导向语言,其主要运行在浏览器环境中,但也支持Node.js等服务器端运行环境。它的语法深受C和Java的影响,但更加简洁和灵活。 1. **变量与数据类型**:JavaScript采用动态...

    javascript精彩技巧集300例下载

    5. **异步编程**:JavaScript使用回调函数、Promise和async/await来处理非阻塞操作,如网络请求。例子可能包括`fetch()`API、`setTimeout()`和`setInterval()`。 6. **数组方法**:`map()`, `filter()`, `reduce()`...

    JavaScript实用技巧集锦

    6. **异步编程**:JavaScript的事件循环和回调函数、Promise、async/await提供了处理异步操作的方式,理解这些机制对于编写高性能的前端应用至关重要。 7. **模板字符串(Template literals)**:ES6引入的新特性,...

    JS的调试技巧

    JavaScript,简称JS,是Web开发中的重要脚本语言,用于实现客户端交互和动态网页效果。掌握有效的JS调试技巧对于开发者来说至关重要,可以帮助我们快速定位并解决问题。以下是一些关于JS调试的关键知识点: 1. **...

Global site tag (gtag.js) - Google Analytics