【JavaScript】变量定义提升、this指针指向、运算符优先级、原型、继承、全局变量污染、对象属性及原型属性优先级
参考资料http://caibaojian.com/toutiao/5446
1、所有变量声明(var)或者声明函数都会被提升到当前函数顶部
关于函数表达式,js会将代码拆分为两行代码分别执行。这里需要注意的是
var getName 和 function getName都是声明语句,区别在于var getName是函数表达式,function getName是函数声明,这里使用var进行函数声明和变量声明时一样的,都是声明提前至当前函数最顶部,二函数声明是将函数声明提升至当前函数顶部。(这个有点绕,但是可以看到本质就是声明提前,function getName整个是函数声明,声明提前,而var getName后接的是函数表达式,和声明变量一样,声明提前,赋值留在原地。)
(实名函数和匿名函数,这个说法可能会更熟悉一些,
实名函数:不论声明在哪里都会在其他代码执行前,被提升到当前作用域最顶端以准备被调用;
匿名函数:(函数的表达式的表达形式)实际上是个变量,在声明之前无法被调用。)
console.log(x);
var x = 1;
function x(){};
这三行简单的代码,var x = 1;将被拆分为var x;x=1;两个部分,声明提前至当前函数顶部,赋值留在原地,而函数表达式也会提升至当前函数最上方。
var x;
function x(){};
console.log(x);
x=1;
实际运行过程中函数如上顺序执行,因此输出x为function(){}。
当没有进行var声明的时候,需要在当前函数作用域寻找是否存在该变量声明,如果没有,则向上一层继续寻找,注意,如果此处依然没有,那么会向上寻找直至window对象,若window对象中也没有该属性,那么就会在window中创建该变量。
2、this指向是由所在函数的调用方式来决定的。
3、关于构造函数
构造函数不能直接被调用,必须通过new运算符在创建对象时才会自动调用,一般当函数执行到的时候自动调用。
4、this在构造函数中代表当前实例化对象。
题目
function Foo(){
getName = function(){alert(1);};
}
Foo.getName = function(){alert(2);};
Foo.prototype.getName = function(){alert(3);};
var getName = function(){alert(4);};
function getName(){alert(5);};
(1)Foo.getName();
(2)getName();
(3)Foo().getName();
(4)getName();
(5)new Foo.getName();
(6)new Foo().getName();
(7)new new Foo().getName();
在理解了上面的话以后前四题应该是非常简单的,归根结底一句话,变量声明和声明函数提前至当前函数顶部。
(1),直接调用Foo.getName(),输出2;
(2),getName(),函数顺序执行,这里我们看到var getName和function getName函数表达式即将var getName 提至顶部,下面将函数声明提前,那么是先运行的function getName(){alert(5);};后执行getName = function(){aleert(4);};后面将前面覆盖了,输出4。
(3),显然先调用Foo函数,第一个调用以后,这里运行了getName = function(){alert(1)}这是一个函数表达式,我们先看当前函数内部是否存在变量声明,并没有那么找到外面,var getName = function(){alert(4);};这里就覆盖了这个函数表达式,进行新的赋值。此处return this,即执行this.getName(),此处直接调用,那么this指向window。输出1.
(4),经过上一步运行,getName()被覆盖,依然输出1。
5~7题
主要是构造函数通过new进行实例化。
(5),new(Foo.getName)(),将getName函数通过构造函数来执行,输出2
这里有一个构造函数的返回值问题。
在js构造函数中可以有返回值也可以没有,我们这样看
function F(){};
>undefined
new F();
>F {}
如果有返回值,检查检查值是否是引用类型,如果是非引用类型,如基本类型(string、number、boolean、null、undefined)则与无返回值相同,实际返回其实例化对象。
如果是引用类型,那么实际返回值为引用类型。
function f(){return true;};
>undefined
new f();
>f {}
function f(){return {1:"a"}};
>undefined
new f();
>Object {1: "a"}
此处返回值为this,this在构造函数中表示当前实例化对象,因此返回当前实例化对象。因为在Foo构造函数中没有为构造函数添加任何属性,因此到当前对象的原型对象(prototype)中寻找getName,找到了。
(6)(7)输出3.
【JavaScript】变量定义提升、this指针指向、运算符优先级、原型、继承、全局变量污染、对象属性及原型属性优先级的更多相关文章
- JavaScript 变量声明提升
JavaScript 变量声明提升 一.变量提升的部分只是变量的声明,赋值语句和可执行的代码逻辑还保持在原地不动 二.在基本的语句(或者说代码块)中(比如:if语句.for语句.while语句.swi ...
- js变量定义提升、this指针指向、运算符优先级、原型、继承、全局变量污染、对象属性及原型属性优先级
原文出自:http://www.cnblogs.com/xxcanghai/p/5189353.html作者:小小沧海 题目如下: function Foo() { getName = functio ...
- javascript 变量定义
一.javascript中,变量定义的位置与写在哪个<script></script>标签对内无关,只分前后顺序,写在前面的后面就能够访问,写在后面的前面会提示“未定义”. 例 ...
- javascript变量声明提升和函数声明提升
在ES6之前,JavaScript没有块级作用域(一对花括号{}即为一个块级作用域),只有全局作用域和函数作用域.变量提升即将变量声明提升到它所在作用域的最开始的部分. JS的解析过程分为两个阶段:预 ...
- 浅谈JavaScript变量声明提升
前段时间阿里实习生内推,一面就被刷了,也是郁闷.今天系统给发通知,大致意思就是内推环节不足以了解彼此,还可以参加笔试,于是赶紧再投一次.官网流程显示笔试时间3月31日,时间快到了,开始刷题.网上搜了一 ...
- JavaScript变量声明提升
JavaScript代码在被解析引擎执行前,会被“编译”把变量声明等放在合适的作用域中,如果不了解这一点,会让人产生很多疑惑. 文章:详解js变量声明提升
- javascript变量声明提升(hoisting)
javascript的变量声明具有hoisting机制,JavaScript引擎在执行的时候,会把所有变量的声明都提升到当前作用域的最前面. 先看一段代码 1 2 3 4 5 var v = &quo ...
- 【JavaScript高级进阶】JavaScript变量/函数提升的细节总结
// 测试1 console.log('----------test1--------------'); console.log(global); // undefined var global = ...
- javascript变量名提升
预解析的过程 代码的执行过程 程序在执行过程,会先将代码读取到内存中检查,会将所有的声明在此时进行标记.所谓的标记就是让js解释器直到有这个名字,后面在使用名字的时候,不会出现未定义的错误,这个标记就 ...
随机推荐
- Match:Power Strings(POJ 2406)
字符串前缀的阶 题目大意:求前缀的阶 和POJ1961是一样的,KMP的Next数组的应用,不要用STL,不要一个一个读入字符(IO永远是最慢的) #include <iostream> ...
- hihocoder 1061.Beautiful String
题目链接:http://hihocoder.com/problemset/problem/1061 题目意思:给出一个不超过10MB长度的字符串,判断是否里面含有一个beautiful strings ...
- js计算在线时长
后台传来毫秒,前台转换成对应的时分秒 //在线时长 function longTime(seconds){ //总秒数 //seconds=seconds/1000; //得到小时 var hour ...
- 【XLL API 函数】xlfUnregister (Form 1)
此函数可以被 Excel 已经载入的 XLL 或 DLL 调用.它等效于宏表函数 UNREGISTER. xlfUnregister 有两种调用形式: 形式1:Unregister 单独的命令或函数 ...
- 警告:Assigning to 'id<Delegate>' from incompatible type 'ViewController *const_st
原因: 你自己写了代理,设置了 delegate = self.但是self 没有遵守这个协议 只需要遵守这个协议就可以消除警告.
- php字符串处理函数相关操作
<?php//获取tech和98426这两个字符串
- 在没装VS2010的机器上运行VS2010开发的C++程序
在VS2010下写了一个win32控制台应用程序,编译ok.exe,需要依赖osg相关动态库 第一次编译的是Debug版本的,直接将ok.exe和osg相关dll文件拷贝到没有安装VS2010机器上运 ...
- [Android] adb shell dumpsys的使用
reference to :http://blog.csdn.net/g19920917/article/details/38032413 有两种方法可以查看service list: 1. adb ...
- MongoDB 基础 -安全性-(权限操作)
和其他所有数据库一样,权限的管理都差不多一样.mongodb存储所有的用户信息在admin 数据库的集合system.users中,保存用户名.密码和数据库信息.mongodb默认不启用授权认证,只要 ...
- Mac怎么读写NTFS格式?Mac读写NTFS格式硬盘教程
我们都知道NTFS 格式的 Windows 硬盘在Mac OS X系统下只能读取不能写入,这一问题一直困扰着很多新老Mac 用户,一般的的解决办法就是安装 NTFS 插件来让 OS X 支持 NTFS ...