简单理解JavaScript中的封装与继承特性
javascript中的封装
封装简单地说就是让外界只能访问对象的共有变量和函数,隐藏细节和数据。
js中有三种方法创建对象,分别为门户大开型、用命名规范区分私有变量、闭包创建真正的私有变量三种。
1.门户大开型,是实现对象的最基础的方法,所有方法与变量都是共有的外界可以访问。
var book = function(name){ if(this.check(name)){ console.log("error"); throw new error("name null"); } this.name = name; } book.prototype = { check:function(name){ if(!name){ return true; } }, getname:function(){ return this.name; } } var book = new book("哈哈"); //output:哈哈 哈哈 console.log(book.name,book.getname());
这个例子是门户大开型的典型,外界能直接访问对象的属性和方法。可以注意到属性和变量都有"this"来创建。
2.用命名规范区分私有变量,该方法是门户大开型的优化版本,只不过是在私有变量或方法前面用"_"区分,如果有程序员有意使用_getname()的方法来调用方法,还是无法阻止的,不是真正地将变量隐藏。
3.闭包创建真正的私有变量,该方法利用js中只有函数具有作用域的特性,在构造函数的作用域中定义相关变量,这些变量可以被定义域该作用域中的所有函数访问。
var book2 = function(name){ if(check(name)){ console.log("error"); throw new error("name null"); } name = name; function check(name){ if(!name){ return true; } } this.getname = function(){ return name; } } book2.prototype = { display:function(){ //无法直接访问name return "display:"+this.getname(); } } var book2 = new book2("哈哈"); //output:undefined "哈哈" "display:哈哈" console.log(book2.name,book2.getname(),book2.display());
可以看到,这个例子中的结果,直接访问name会返回undefined的结果。可以看到这个例子与门户大开型的区别,门户大开型中的变量使用"this"来创建,而这个例子中使用var来创建,check函数也是如此,使得name与check函数只能在构造函数的作用域中访问,外界无法直接访问。
该方法解决了前两种方法的问题,但是也有一定的弊端。在门户大开型对象创建模式中,所有方法都创建在原型对象中,因此不管生成多少对象实例,这些方法在内存中只存在一份,而采用该方法,每生成一个新的对象都会为每个私有变量和方法创建一个新的副本,故会耗费更多的内存。
javascript中的继承
book基类:
var book = function(name){ if(this.check(name)){ console.log("error"); throw new error("name null"); } this.name = name; } book.prototype = { check:function(name){ if(!name){ return true; } }, getname:function(){ return this.name; } }
继承方法:
function extend(subclz,superclz){ var f = function(){} f.prototype = superclz.prototype; subclz.prototype = new f(); subclz.prototype.constructor = subclz; subclz.superclass = superclz.prototype; if(superclz.prototype.constructor == object.prototype.constructor){ superclz.prototype.constructor = superclz; }
使用空函数f作为桥接,可以避免直接实例化父类时调用父类的构造函数带来额外开销,而且当父类的构造函数有参数时,想直接通过subclass.prototype = new superclass();实现父类构造函数的调用和原型链的继承是不行的。
subclz.superclass = superclz.prototype; if(superclz.prototype.constructor == object.prototype.constructor){ superclz.prototype.constructor = superclz; }
添加这三句可以避免子类继承父类写book.call(this,name);而是简单地写artbook.superclass.constructor.call(this,name)便能实现。
并且在子类重写父类方法的时候,可以调用到父类的方法:
artbook.prototype.getname = functiion(){ return artbook.superclass.getname.call(this) + "!!!"; }
artbook子类:
var artbook = function(name,price){ artbook.superclass.constructor.call(this,name); this.price = price; } extend(artbook,book); artbook.prototype.getprice = function(){ return this.price; } artbook.prototype.getname = function(){ return artbook.superclass.getname.call(this)+"!!!"; }
推荐阅读
-
简单理解JavaScript中的封装与继承特性
-
深入理解JavaScript 中的匿名函数((function() {})();)与变量的作用域
-
详解JavaScript中基于原型prototype的继承特性
-
Javascript中的prototype与继承
-
深入理解JavaScript编程中的同步与异步机制
-
理解 javascript 中的函数表达式与函数声明
-
JavaScript继承的特性与实践应用深入详解
-
JavaScript中数组继承的简单示例
-
JavaScript中变量、常量与字面量的快速理解
-
PHP中的call_user_func()与call_user_func_array()简单理解