JavaScript预解析及相关技巧分析
作者:zhangw428 发布时间:2024-04-10 10:57:32
本文实例讲述了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程序设计有所帮助。


猜你喜欢
- 递归函数两大特点:1.能够调用函数自身2.至少有一个出口(结束函数自身调用)函数实现:def calnum(num): if n
- 第一步:要使用vant组件安装好vant,npm i vant -S第二步:在你要用到的地方js中引入或者在src/main.js里面引入i
- 我们一般在Excel里面是使用数据连接属性里面写sql语句,或者vba里面利用ado组件执行sql语句。新版的Excel里面带上了Power
- 翻译:用法:zeros(shape, dtype=float, order='C')返回:返回来一个给定形状和类型的用0填充
- 本文实例讲述了golang实现的文件上传与文件下载功能。分享给大家供大家参考,具体如下:upload.gopackage commonimp
- 临近下班的时候,突然想到统计热门文章的问题。以前我所知道的热门文章统计,基本有这么几种:按点击数排序 该方法最大的问题在于热门的文章会越来越
- 背景是:在实际开发中,可能会遇到网络问题或者查询量比较大的情况,上一个请求还没有完成,用户就发起了下一个请求。会造成什么情况呢?但是同一个请
- 获取百度的歌曲名,歌手和链接!! package webTools; import java.io.BufferedReader; impo
- CSS是制作网页效果必不可少的东西,字体的颜色定义、表格的样式定义、图片的特效等等都少不了它。但在Dr
- 下面给大家介绍远程连接mysql错误代码1130的解决方法:以上所述是小编给大家介绍的远程连接mysql错误代码1130的解决方法网站的支持
- 一、前言我想介绍这些功能的主要原因是它们可以帮助您避免编写循环。在某些情况下,循环运行可能会很昂贵,除此之外,这些功能将有助于提高速度。以下
- 在cmd控制台内,vue -V 可看到vue-cli脚手架的版本号,现在好多帖子误写成vue版本号。如下图:vue版本号在项目中,找到pac
- 在SQL SERVER中,你可能需要获得当前日期和计算一些其他的日期,例如,你的程序可能需要判断一个月的第一天或者最后一天。你们大部分人大概
- Python 多进程和数据传递的理解python不仅线程用的是系统原生线程,进程也是用的原生进程进程的用法和线程大同小异import mul
- Python自动化测试-使用Pandas来高效处理测试数据一、思考1.Pandas是什么?功能极其强大的数据分析库可以高效地操作各种数据集c
- 用asp程序进行网页设计,大多因为需要访问数据库,然后再将数据显示到页面,如果数据很多的话,页面的访问速度也就变慢了,为了解决这个问题,可以
- 问题我试图打印some_cell.font.color.rgb并得到各种结果。对于一些人,我得到了我想要的东西(比如“ FF000000”)
- php获取域名的google收录示例function get_index($domain){ $url="http://www.g
- 科学设计你的网站网页:来自 Eye-Tracking研究的23节必修课 ——Christina Laun在网络设计领域关于Eye-
- 本文实例为大家分享CentOS 7.2 Yum安装mysql5.6的方法,供大家参考,具体内容如下配置CentOS SCLo源[3] 添加