JavaScript预解析及相关技巧分析


Posted in Javascript onApril 21, 2016

本文实例讲述了JavaScript预解析及相关技巧。分享给大家供大家参考,具体如下:

变量

同样,以这两个小例子的错误对比提示开始。

alert(y1);     //代码段1
var y1 = 'dddd';
alert(y2);     //代码段2 // alert(typeof y2);
y2 = 'xxxxx';

先思考一下,为什么一个会提示 undefined , 一个却抛出变量未定义的错。。先看JavaScript的解析过程。

javascript在执行过程之前,会做一件事件“预解析”。解析引擎会以块为单位级别,执行所有 var 变量的创建,并赐予它们一个初始值:undefined。这样一来,第一个例子为什么会弹出undefined的原因就显而易见了。

所以第一段代码其实就等价于

var y1;
alert(typeof y1); //很自然,它此时值为undefined
y1 = 'dddd';

那第二段代码为什么又抛错呢,此时就不再属于“预解析”阶段了,(在这里我假设浏览器遇到一个script标签时只做两件事:预解析和执行,其实不 只这两件事情),而处于执行阶段,抛错原因在于js在执行段状态并不知晓y2的状态(预解析阶段没有捕获到y2的任何信息),当然就抛出未定义的错误信 息。这里又牵扯到另一个问题: js 是弱类型语言,变量不定义可以用,那为什么这里又抛出为定义错误呢。

事出总有因,javascript总有很多自身奇怪的特性,有一个叫变量的读写不均等。未定义的变量,仅仅是可写,并不可读。何为可写,每个人都熟悉这样的写法:

y2 = 'exam';  
//在没出现它的定义操作之前(即在它还没有自己的scope之前)该操作会将这段代码认为是定义一个全局变量,在window上注册一个属性y2,并赋值为exam

但在对它的读操作,js引擎搜寻不到它的任何相关信息,就以自己的脾气行事,毫不客气抛一个未定义的错,这是js的游戏规则。还有但是,那为什么却 可以获取它的类型呢。还记得 js对对象的操作吧。如果访问对象一个不存在的属性及类型,会提示undefined, (因为它当前是window对象的一个属性)。

再注:这里需要区分, 变量的读写不均等 仅用于变量,读取所有对象的属性,不存在该特性,如果不存在,会提示undefined。

结论

到这里,我的思考结果: 对于变量和对象的写操作,他们有一定的类似之处.但是读操作,各有各的一套规则,正因为此,所以有了上面的问题。
这样一来,下面这个题应该就很容易得到答案了吧。

if (!('a' in window)) {
  var a = 1;
}
alert(a);

函数

引申一下,function. 还记得上面提到的预解析,在javascript的预解析中,除了对var 变量的预定义,还包括了提取对函数的定义,所以可以在script的任何地方定义函数,在任何地方调用。不限于它之前.

但函数的定义方式,包括了一种叫字面量定义法, 用var的方法声明function.看下面

alert(typeof y3); //结果?
var y3 = function (){ console.log('1'); }

还记得这个约定吧:调用必须出现在声明之后,为什么呢,如果理解了上面,其实这里答案已经明了。javascript 引擎在预解析 var 时 会给他们一个初始值 undefined,这样一来,如果我们在它的声明之前调用它,javascript 引擎还没拿到它的真实值,自然会报"xxx is not a function" 的错.这也理清了为什么同为函数声明,一个却关系到声明和调用的顺序,一个却无这样的约束。

结论

它是函数,是js执行,动态修改的结果,依然遵循了变量的预解析规则(在上面alert的时候,它还并没有拿到字面量函数的信息)。
如果是两个混合呢。看下面, 同时存在了为y4的变量和function。

alert(typeof y4); //结果?
function y4(){
  console.log('y4')
}
var y4;

因为 javascript 在预解析时function的声明优先级高的缘故,所以y4自然为function类型, 但是在当y4 赋值之后(此时js引擎处于执行过程中),它对js的赋值操作将会覆盖function的声明。所以:

alert(typeof y5);
var y5 = 'angle';
function y5(){
  console.log('ghost');  
}
alert(y5);

第一次alert结果,因为它处于js 执行过程中的顶端,所以为 function。 第二次再alert时, 它的值已经被重写为5(不要被function的定义位置在下所迷惑。)

从js的解析和执行分开来想,才发现眼前豁然开朗,很多问题的答案都很自然得浮出水面,正如那篇文章作者所说,"一旦理解了执行环境、调用对象、闭包、词法作用域、作用域链这些概念,JS语言的很多现象都能迎刃而解。"

现在再看,即使在这个存在着很多不可思议的语言中,也有很多可以追溯到它的缘由。

如何更好的做参数判断

讨论了上面那么多,那如何让它更贴近于实际开发呢,既然javascript的读写不均等性,如何才能避免在不报错的情况下做参数判断呢。

eg:

if(cusVar){ //这里的判断,是不是存在隐含的问题呢。 }

如何严谨一些呢。

if(window['cusVar']) { //保证它不报错。
  //或者这样的判断也是可行的 window.cusVar | typeof cusVar !== 'undefined'
  //干活咯
}

最后补充又一个小quiz, (理解 预解析与执行的分离)

var y7 = 'test';
function fun1(){
  alert(y7);
  var y7 = 'sex';
}
fun1();

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
使用dynatrace-ajax跟踪JavaScript的性能
Apr 12 Javascript
jquery教程ajax请求json数据示例
Jan 13 Javascript
JavaScript运行机制之事件循环(Event Loop)详解
Oct 10 Javascript
AngularJs实现ng1.3+表单验证
Dec 10 Javascript
JS实现对中文字符串进行utf-8的Base64编码的方法(使其与Java编码相同)
Jun 21 Javascript
jquery动态遍历Json对象的属性和值的方法
Jul 27 Javascript
Javascript实现前端简单的路由实例
Sep 11 Javascript
Angular中$state.go页面跳转并传递参数的方法
May 09 Javascript
微信小程序实现action-sheet弹出底部菜单功能【附源码下载】
Dec 09 Javascript
es6新特性之 class 基本用法解析
May 05 Javascript
在微信小程序中使用图表的方法示例
Apr 25 Javascript
详解json串反转义(消除反斜杠)
Aug 12 Javascript
JavaScript实现显示函数调用堆栈的方法
Apr 21 #Javascript
JS实现队列与堆栈的方法
Apr 21 #Javascript
JS、jQuery中select的用法详解
Apr 21 #Javascript
jQuery中iframe的操作(点击按钮新增窗口)
Apr 20 #Javascript
JS生成某个范围的随机数【四种情况详解】
Apr 20 #Javascript
浅析JS动态创建元素【两种方法】
Apr 20 #Javascript
纯JS代码实现一键分享功能
Apr 20 #Javascript
You might like
默默小谈PHP&MYSQL分页原理及实现
2007/01/02 PHP
php 字符转义 注意事项
2009/05/27 PHP
thinkPHP5 tablib标签库自定义方法详解
2017/05/10 PHP
laravel 错误处理,接口错误返回json代码
2019/10/25 PHP
让回调函数 showResponse 也带上参数的代码
2007/08/13 Javascript
javascript实现日历控件(年月日关闭按钮)
2012/12/12 Javascript
jquery获取所有选中的checkbox实现代码
2016/05/26 Javascript
BootStrap智能表单demo示例详解
2016/06/13 Javascript
JavaScript 函数的定义-调用、注意事项
2017/04/16 Javascript
swiper自定义分页器使用方法详解
2020/09/14 Javascript
Vue 仿QQ左滑删除组件功能
2018/03/12 Javascript
几个你不知道的技巧助你写出更优雅的vue.js代码
2018/06/11 Javascript
vue模块拖拽实现示例代码
2019/03/09 Javascript
使用layer弹窗,制作编辑User信息页面的方法
2019/09/27 Javascript
微信小程序后端实现授权登录
2020/02/24 Javascript
vue移动端弹起蒙层滑动禁止底部滑动操作
2020/07/22 Javascript
three.js 如何制作魔方
2020/07/31 Javascript
Python爬虫利用cookie实现模拟登陆实例详解
2017/01/12 Python
使用python实现BLAST
2018/02/12 Python
Python GUI布局尺寸适配方法
2018/10/11 Python
深入浅析Python2.x和3.x版本的主要区别
2018/11/30 Python
Tensorflow 使用pb文件保存(恢复)模型计算图和参数实例详解
2020/02/11 Python
python爬虫开发之Request模块从安装到详细使用方法与实例全解
2020/03/09 Python
简单总结CSS3中视窗单位Viewport的常见用法
2016/02/04 HTML / CSS
ziaja齐叶雅官方海外旗舰店:来自波兰的天然护肤品牌
2017/01/02 全球购物
Janie and Jack美国官网:GAP旗下的高档童装品牌
2019/09/09 全球购物
简述DNS进行域名解析的过程
2013/12/02 面试题
深入开展党的群众路线教育实践活动方案
2014/02/04 职场文书
《蒙娜丽莎之约》教学反思
2014/02/27 职场文书
《春到梅花山》教学反思
2014/04/16 职场文书
重阳节活动总结
2014/08/27 职场文书
2014年幼儿园教师工作总结
2014/11/08 职场文书
晚自修旷课检讨书怎么写
2014/11/17 职场文书
golang三种设计模式之简单工厂、方法工厂和抽象工厂
2022/04/10 Golang
MySQL安装失败的原因及解决步骤
2022/06/14 MySQL
TypeScript实用技巧 Nominal Typing名义类型详解
2022/09/23 Javascript