1.变量提升
JavaScript的函数定义有个特点,它会先扫描整个函数体的语句,把所有申明的变量“提升”到函数顶部:
1 | ; |
对于上述foo()
函数,JavaScript引擎看到的代码相当于:
1 | function foo() { |
养成规范:所有 的变量定义都放在函数的头部,不要乱放,便于代码维护。
1 | function foo() { |
2.全局变量
变量在函数外定义,即为全局变量。
全局变量有 全局作用域: 网页中所有脚本和函数均可使用。
1 | var x = 2;// 全局变量 |
变量生命周期:
JavaScript 变量生命周期在它声明时初始化。
局部变量在函数执行完毕后销毁。
全局变量在页面关闭后销毁。
全局对象window
在 HTML 中, 全局变量是 window 对象: 所有数据变量都属于 window 对象。
1 | ; |
alert() 这个函数本身也是一个window的变量。
JavaScript实际上只有一个全局作用域。任何变量(函数也视为变量),如果没有在当前函数作用域中找到,就会继续往上查找,最后如果在全局作用域中也没有找到,则报ReferenceError
错误。
3.名字空间
全局变量会绑定到window
上,不同的JavaScript文件如果使用了相同的全局变量,或者定义了相同名字的顶层函数,都会造成命名冲突,并且很难被发现。
如何解决冲突?
把自己的所有变量和函数全部绑定到一个全局变量中。
1 | // 唯一的全局变量MYAPP:也就是一个空对象 |
也就引入了名字空间,类似C++中的namespace。
4.局部作用域
由于JavaScript的var
关键字申明的变量作用域实际上是函数内部,我们在for
循环等语句块中是无法定义具有局部作用域的变量的。
所以ES6引入了新的关键字let
,用let
替代var
可以申明一个块级作用域的变量。
块级作用域也就是一个{}
的范围。
建议:只在开一个全局变量的名字空间时使用var
,其他情况使用let
关键字。
5.常量
ES6标准引入了新的关键字const
来定义常量,const
与let
都具有块级作用域:
1 | ; |
6.方法
在一个对象中绑定函数,称为这个对象的方法。
1 | var MYAPP = {}; |
this
关键字的作用?
1 | var MYAPP = {}; |
这个this
的指向是JS中的一大坑!!!
更坑爹的是,如果这么写:
1 | var fn = xiaoming.age; // 先拿到xiaoming的age函数 |
也是不行的!要保证this
指向正确,必须用obj.xxx()
的形式调用!
ECMA决定,在strict模式下让函数的this
指向undefined
,因此,在strict模式下,你会得到一个错误。
7.控制this的指向
在JS中,我们可以控制this的指向。
要指定函数的this
指向哪个对象,可以用函数本身的apply
方法,它接收两个参数,第一个参数就是需要绑定的this
变量,第二个参数是Array
,表示函数本身的参数。
用apply
修复getAge()
调用:
1 | function getAge() { |
8.常用内部对象
8.1 标准对象
我们用typeof
操作符获取对象的类型,它总是返回一个字符串:
1 | typeof 123; // 'number' |
特别注意null
的类型是object
,Array
的类型也是object
,如果我们用typeof
将无法区分出null
、Array
和通常意义上的object——{}
。
8.2 Date
1 | var now = new Date(); |
注意,当前时间是浏览器从本机操作系统获取的时间,所以不一定准确,因为用户可以把当前时间设定为任何值。
你可能观察到了一个非常非常坑爹的地方,就是JavaScript的月份范围用整数表示是0~11,0
表示一月,1
表示二月……,所以要表示6月,我们传入的是5
!
时间戳还原成标准时间:
1 | var t = new Date(); |