组合继承的问题是会调用2次超类型构造函数
寄生组合式继承
即通过借用构造函数来继承属性,通过原型链的形式来继承方法,思路:不必为了指定子类型的原型而调用超类型的原型,我们所需要的无非是超类型原型的一个副本而已
function inheritPrototype(subType,superType){
var prototype=object(subType.prototype); //创建对象
prototype.constructor=subType; //增强对象
subType.prototype=prototype; //指定对象
}
第7章 函数表达式
定义函数有两种方式:函数声明和函数表达式
函数声明 function functionName(arg0,arg1,arg2){
//函数体
}
函数声明提前:执行代码前会先读取函数声明
函数表达式 var functionName=function(arg0,arg1,arg2){
//函数体
};
由于函数声明提前的存在,最好在函数体内使用函数表达式而非函数声明
递归函数就是一个函数通过名字调用自身的情况下构成的
arguments.callee指向一个正在执行的函数的指针,可以用它来实现对函数的递归调用
闭包是指有权访问另一个函数作用域中的变量的函数
作用域链的概念:当某个函数第一次被调用,会创建一个执行环境及相应的作用域链,并把作用域链赋值给一个特殊的内部属性(即[[Scope]])。然后,使用this、arguments和其它参数的值来初始化函数的活动对象。但在作用域链中,外部的函数活动对象始终处于第二位,以此类推直到作用域链的终点为全局执行环境
作用域链本质上是一个指向变量对象的指针列表,它只引用但不实际包含变量对象
由于闭包会携带包含它的函数的作用域,因此会占用更多内存,需要谨慎使用
闭包的这种配置机制引出了一个问题,即闭包只能取得包含函数中任何变量的最后一个值,我们可以创建另一个匿名函数强制让闭包的行为符合预期
在闭包中使用this可能会导致一些问题,我们知道,this对象在运行时是基于函数的执行环境绑定的:在全局函数中,this等于window,而当函数被作为某个对象的方法调用时,this等于那个对象,匿名函数的执行环境具有全局性,因此其this对象通常指向window
内存泄漏:闭包会引用包含函数的整个活动对象,把变量设置为null,可以解除引用,回收其占用的内存
模仿块级作用域:JavaScript中没有块级作用域规定,只要变量定义在一个函数的活动对象中,就可以在函数内部随处访问它
(function(){
//这里是块级作用域
}) ();
特权方法:我们把有权访问私有变量和私有函数的公有方法称为特权方法,在函数中定义特权方法有一个缺点,那就是你必须使用构造函数模式来达到这个目的,第6章提到过,构造函数会重复创建一组新方法
静态私有变量:在之前的模拟块级作用域中定义一个全局构造函数(使用函数表达式并不使用var),然后在这个构造函数的原型中定义特权方法,这样所有实例共享一个特权方法,对原型的修改也会反映到所有实例
模块模式
模块模式则是为单例创建私有变量和特权方法,所谓单例就是只有一个实例的对象,声明并赋值一个函数给变量,这个函数里拥有私有变量和函数,return对象字面量(这个字面量对象可以访问到私有变量和私有函数)
简言之,如果必须创建一个对象并以某些数据对其进行初始化,同时还要公开一些能够访问这些私有数据的方法,那么就可以使用模块模式,以这种方式创建的单例都是Object的实例
增强的模块模式
进一步改进了模块模式,在返回对象之前加入对其增强的代码
JavaScript高级程序设计15.pdf,布布扣,bubuko.com