1、类型转换
-
原始值到对象的转换
通过调用String()
、Number()
、Boolean()
构造函数,转换为对应的包装对象。null
和undefined
没有包装对象,所以在期望是对象的地方不会执行正常的隐式转换,抛出异常。 -
显式类型转换和隐式类型转换
显式类型:使用String()
、Number()
、Boolean()
或Object()
函数作为类型转换函数;类似于其他语言中的强制类型转换。
Number("3") //=> 3
String(false) //=> "false"或使用false.toString(),除了null和undefined之外的任何值都有toString()方法
Boolean([]) //=>true (除了假值,其他任何值都是真值)
Object(3) //=>new Number(3)
Object(null) //=>返回一个新创建的空对象,不会像隐式转换那样报错,undefined同样如此
隐式类型:某些运算符会做隐式的类型转换。比如,
- 若二元“+”运算符的一个操作数是字符串,它将会把另一个操作数转换为字符串;
- 一元“+”运算符将其操作数转换为数字;
- 一元“!”运算符将其操作数转换为布尔值再取反。
- 比较远算符和“==”也会对不同情况进行相应的类型 转换,比较运算符偏爱数字,只要其中一个操作数为数字,则将另一个操作数转换为数字。
-
对象转换为原始值
所有对象到布尔值的转换均为true
。
对象到字符串和数字则通过待转换对象的一个方法来完成。
所有的对象继承了toString()和valueOf()方法。通过使用toString()和valueOf()方法完成对象到字符串和对象到数字的转换。根据自己的理解总结如下:
-
普通的对象的toString()方法将返回对象的类属性(是一个字符串,用于表示对象的类型信息);
eg. ({x:1,y:2}).toString() //=>"[object Object]"
-
但是很多类重新定义了toString()方法:
1> 数组类的toString()方法将每个数组元素转换为一个字符串,并在元素之间添加逗号合并成结果字符串;
2> 函数类的toString()方法返回这个函数的实现定义的表示方式(即js源代码字符串);
3> 日期类返回一个可读的日期和时间字符串;
4> RegExp类将RegExp对象转换为表示正则表达式直接量的字符串;
各示例如下:
[1,2,3].toString() //=>"1,2,3"
(function(x) { f(x); }).toString() //=> "function(x) {\n f(x);\n}"
/\d+\g.toString() //=>"/\\d+/g"
new Date().toString() //=>"Tue Apr 25 2017 17:40:05 GMT+0800 (中国标准时间)"
- 默认的valueOf()方法简单地返回对象本身。数组、函数和正则表达式简单地继承了这个默认方法。日期类定义的valueOf()方法返回它的一个内部表示:1970年1月1日以来的毫秒数。
当运算符“+”、“==”、“!=”和关系运算符其中一个操作数是对象时,JavaScript将隐式转换对象到原始值。
2、变量声明
- 使用var重复声明变量合法,如果带有初始化量相当于一条简单的赋值语句;
- 在函数体内,局部变量的优先级高于同名的全局变量;
- JavaScript会将变量声明提前,只有在程序执行到var语句的时候,局部变量才会被真正的赋值;见如下示例:
function f(){
console.log(scope); //=>"undefined"
var scope="local";
console.log(scope); //=>"local"
}
- 当使用var声明一个变量时,创建的这个属性时不可配置的,也就是说这个变量无法通过delete运算符删除;
3、作用域链
当JavaScript需要查找变量x的时候(这个过程叫做“变量解析”),它会从作用域链中的第一个对象开始查找,若第一个对象中不存在名为x的属性,则会继续查找链上的下一个对象。若作用域链上没有任何一个对象含有属性x,则抛出异常(ReferenceError)。
例如,在不包含嵌套的函数体内,作用域上有两个对象,第一个是定义函数参数和局部变量的对象,第二个是全局变量。可以理解为从最内层往外扩展。
网友评论