一、语言基础
1.javascript中,只有null和undefined是无法拥有方法的值,它们都没有包装对象。typeof null == ‘object' , typeof undefined =='undefined'
2.在javascript运算中,上溢返回infinity,下溢返回0,也就是说被0整除并不报错,而是返回无穷大。而0/0返回NaN。
3.NaN的特殊性,它和任何值都不相等,也就是说,没法通过x==NaN来判断x是否是NaN,只能通过x!=x来判断,函数isNaN()的作用相同。
4.任意Javascript的值都可以转换为布尔值。
5.原始值是不可变的,它们的比较是值的比较,只有它们的值相等时才相等。对象时可变的,对象的比较并非值的比较,而是引用的比较,只有当它们引用同一个对象的时候,才相等。
6.if语句将undefined转换为false,但==从不试图将操作数转换成布尔值。
7.从数字到字符串的准换有三种方法,tofixed()指定小数位数,toExponential()指定指数计数法中小数点后的位数,toPrecision()指定有效数字位数。
8.对象转换为字符串,首先使用toString()方法,如果不存在或者不返回原始值,则调用valueOf()方法;对象转换为数字,首先尝试使用valueof()方法,没有才使用toString()。
9.变量声明提前中吗,只有执行到var语句时,局部变量才会真正被赋值。
10.通过var声明的全局变量与隐式声明的全局变量区别为,通过var声明的变量不可删除,而隐式声明的变量可以删除。
11.函数声明语句与函数定义表达式的区别为:使用var,只有变量名提前,但是变量的初始化部分还在原来的位置,而使用函数生命的话,函数名称和函数体均提前。
12.在switch case中每个case的匹配操作实际是===而不是==。
13.在for..in循环中,并不会遍历对象的所有属性,而是只有可枚举属性,可枚举属包括了自定义属性和方法,以及继承的自定义属性。不可枚举属性是指语言核心所定义的内置方法和属性。
14.return可以单独使用,这样的话是返回undefined。
15.debugger语句,用来产生一个断点,但是它不会自动启动调试器,只有调试器已经运行,才会产生一个断点。
二、use strict
use strict是一条指令而不是一个语句。
严格模式的限制:
1.禁止使用with。
2.所有变量都必须先声明,如果未声明,将抛出异常。
3.在严格模式中,调用函数中的this值为undefined,在非严格模式中,this为全局对象。可以使用这种方式判定javascript是否支持严格模式:
var hasStrictMode = (function(){"use strict " return this === undefined}())
4.在 严格模式中,给只读属性或者不可扩展对象创建新成员都将抛出异常,而在非严格模式中,只会失败,不会报错。
三、对象
1.创建对象
有两种方式,对象直接量和构造函数、。
对象直接量:
var obj ={
name : "jim",
age : 12
}
构造函数创建:
var o =new Object()
2.每一个对象都从原型中继承属性,没有原型的对象不多,Object.prototype就是其中之一。
3.Object.create()是ECMAscript5中定义的方法,它创建一个新的对象,第一个参数是对象哦原型,第二个参数为可选,用于对对象的属性进行进一步描述。
var obj = Object.create({x:1,y:2});
可以通过传入参数null来创建一个没有原型的新对象,单通过这个方式创建的对象不会继承热你和东西,甚至不包括基础方法。
如果想创建一个普通哦空对象,比如通过{}或new Object()创建的对象,需传入Object.prototype。
在ECMAscript3中,可以使用如下方法来模拟Object.create()
function inherit(p){
if(Object.create){
return Object.create(p);
}
var t=typeof p;
if( t !== 'object' && t!== 'function'){
throw TypeError();
}
function f();
f.prototype=p;
return new f();
}
4.属性的查询和设置
在javascript中,只有查询属性的时候,才会体会到继承的存在,而设置属性则和继承无关。
如果访问不存在的属性,则返回undefined,但如果对象不存在,就会报错。
所以要判读属性的存在,需要使用如下方式:
var len = undefined;
if(book){
if(book.title){
len=book.title.length;
}
}
或者
var len = book && book.title&& book.title.length;
5.delete 只能删除自有属性,不能删除继承属性。
6.使用hasOwnProperty()来检查给定的名字是否是对象的自有属性。
7.在ECMAscript5中,属性值可以用setter和getter代替,,由setter和getter定义的属性称为存取器属性(accessor property )。它不同于数据属性(data property)。
和数据属性不同,存取器属性不具有可写性,如果属性同时具有setter和getter方法,那么它是一个可读写属性,如果它只有setter方法,它就是一个只写属性,读取的话返回undefined,如果只有getter,则为只读属性。
定义存取器属性的方法:
var o = {
data:value,
get accessor_prop(),
set accessor_porp(),
}
存取器属性也可以继承。
8.属性的特性,数据属性的特性分别为value、writable、enumerable、configurable。而存取器属性的特性为get、set、enumerable、configurable。
通过调用Object.getOwnPropertyDescriptor(obj,prop)可以获取某个对象特定属性的属性描述符。
如果要设置属性特性,可以使用Object.defineProperty(obj,prop,property);
如果要同时修改多个属性,则需要使用Object.defineProperties(obj,map);
9.在ECMAscript5中,可以使用Object.getPrototypeOf(来查询它的原型。但在ECMAscript3中,则没有与之等价的函数,这时,需要使用o.contructor.prototype来检测一个对象的原型。
要想检查一个对象是否是另外一个对象的原型,请使用isPrototypeOf()方法。
10.对象的类属性是一个字符串,用以表示对象的类型信息,要想获取对象的类,可以使用toString方法,但是很多对象继承的toString()方法都进行了重写,为了能调用正确的toString版本,需要间接调用Function.call()方法。
function classof(o) {
if (o === null) {return "NULL";}if (o === undefined) {return "undefined";}return Object.prototype.toString.call(o).slice(8,-1);}对于自定义的类,没办法通过类属性来区分对象的类。
12.序列化对象,SCMAscript5中提供了JSON.stringify和JSON.parse()方法来序列化和还原对象。函数、RegExp、Error对象和undefined值不能序列化和和还原。JSON.stringify只能序列化对象可枚举的自有属性,对于一个不能序列化的属性,序列化后输出的字符串中会将这个属性省略掉。
四、数组
1.如果省略数组直接量中的某个值,省略的元素将被赋值为undefined,数组直接量允许有可选哦结尾逗号,故[,,]只有连个元素,而非三个。
2.稀疏数组中,length属性大于元素的个数。如果从数组中删除一个元素,它会变成稀疏数组。
3.ECMAscript5中的数组方法包括forEach()、filter()、map()、every()、some()、reduce()、reduceRight()、indexOf()、lastindexOf()。
4.检查数组类型,ECMAscript5中,有Array.isArray()方法。而ECMAscript3中,使用如下方法。
var isArray = Function.isArray() || function(o) {
return typeof o ==="object" &&
Object.prototype.toString.call(o) === "[object Array]";
};
五、函数
1.函数声明语句并非真正的语句,SCMAscript规范只允许它们作为*语句,它们可以出现在全局代码中,或者内嵌在其他函数中,但是不能出现在循环、条件判断、或者try、catch、finally以及with语句中,注意,此限制仅限于以语句声明形式定义的函数,函数定义表达式可以出现在javascript代码的任何地方。
2.通过this来判断当前是否为严格模式:
var strict = (function () { return !this;}());
3.this是一个关键字、表示变量,也不是属性名,javascript语法不允许给this赋值。this没有作用域的限制,嵌套的函数不会从调用它的函数中继承this。如果嵌套函数作为方法调用,this指向调用它的对象,如果嵌套函数作为函数调用,this指向全局对象或者undefined(严格模式)。如果你想访问这个外部函数的this值,需要将this的值保存到一个变量中,这个变量和内部函数都在同一个作用域内。arguments与this类似。
4.如果构造函数没有形参,javascript构造函数调用是允许省略实参列表和圆括号的。
5.构造函数通常不使用return关键字,构造函数表达式的计算结果就是这个新的对象的值,如果构造函数显示返回一个对象,那么调用构造函数的结构就是这个返回的对象,如果使用了return但是没有返回值,或者返回一个原始值,那么将忽略原始值,同时使用这个新的对象作为调用结果。
6.实参对象的属性callee指向当前正在执行的函数,在ECMAscript中禁止使用。
7.自定义函数属性,当函数需要一个静态变量来保持某个值不变的时候,最方便的方式给涵涵素定义属性,而不是定义全局变量,缺点是,这个属性可能被人修改。
8.关联到闭包的作用域都是活动的。
9.在函数中,实参个数通过arguments.length获取,而形参个数通过函数的length属性获取。
10.call()和apply()的区别,call()第一个参数之后的所有参数是要传入的实参,而apply将所有要传入的实参都放在一个数组当中。
11.ECMAscript5中有一个bind方法,可以将某个函数绑定到某个对象,ECMAscript中可以轻松实现这个方法。
function bind(f,o) {
if(f.bind) {return f.bind(o);}else {return f.apply(o, arguments);}}
12.ECMAscript5中,bind不仅仅是将函数绑定到一个对象,还可以实现curry柯里化,除了第一个实参外,其传入bind的实参也会绑定到this。
13.Function()构造函数,所创建的函数并不是使用词法作用域,相反,函数体代码的编译总是顶层函数执行。例如
var scope = “global"
function constructFunction() {
var scope = "local";
return new Function("return scope"); //无法补货局部作用域
}
constructFunction(); //=>global
14.所有函数都是可以调用的,但是不是所有可调用对象都是函数。