由于javascript原生是不支持类的(ES6已经支持class与extends),更不用谈继承、多态了,为了模拟出一些其它面向对象编程语言的这些特性,有好多大牛写了给出了实现方式,看了John Resig的《Simple JavaScript Inheritance》这篇文章,深深被折服了,原来短短几十行javascript也可以这么强大、优雅,下面以我的理解方式来解读下。
主要实现了继承、访问父类的重名方法(这里的实现方式太妙了),但遗憾的是不能实现成员变量/函数的隐藏。
(function(){ //设置标志位,是new A()过程中还是 B=A.extends({/* */})过程中; var initializing = false, //fnTest 可取结果为俩正则对象 /\b_super\b/与 /.*/ //当正则的test方法参数支持自动调用toString()方法时取前面那个 fnTest = /xyz/.test(function(){xyz;}) ? /\b_super\b/ : /.*/; // 创建一个全局的 Class对象 this.Class = function(){}; // 创建继承函数 Class.extend = function(prop) { //此处把_super指向父类的prototype,属性继承时需要用其判断父、子是否有同名方法 var _super = this.prototype; //开始B=A.extends({/* */}); initializing = true; //实例化父类,并把父类的实例方法及属性给prototype var prototype = new this(); initializing = false; //结束B=A.extends({/* */}); // 遍历用户传入的用于构建子类的对象 //处理的地方包括: //1、属性直接存到prototype上 //2、方法中没通过this._super()调用父类中的同名方法,则直接把该方法存到prototype上 //3、方法中有通过this._super()调用父类中的同名方法,则把如下过程包裹成一个函数存到prototype上: // 把this._super指向父类的同名方法,然后再调用子类的该方法并返回执行结果 for (var name in prop) { // 循环体中看起来略微复杂,是整个代码的精华所在 //简化下此过程就是 v = a && b && c ? d : e; 等同于 v = (a && b && c) ? d : e //该过程中主要运用的就是逻辑运算中的短路运算 a,b,c全为true则v=d,否则v=e //a中判断prop[name]是否是函数,b中判断父类中是否也有同名的name函数,c主要判断name函数中是否有调用父类的同名方法(即调用了this._super()) //d就是a,b,c同时满足的时候,也就是说:name是函数,且name函数存在与父类中,且子类的name函数需要调用了父类的同名函数 //若a,b,c中有一项不满足则直接把prop[name]给prototype[name] prototype[name] = typeof prop[name] == "function" && typeof _super[name] == "function" && fnTest.test(prop[name]) ? (function(name, fn){ //首先,先决条件决定了prototype[name]是个函数,所以先包裹一个函数返回 return function() { //作者这里备份,然后调用fn后又还原this._super //由于return的是一个function,具有延时调用的作用 所以在子类调用fn时this始终指向子类本身 //而这里的_super是父类的_super,与this._super并无关系,所以备份应该是多余的 //var tmp = this._super; //this._super指向与fn(即prop[name])同名的父类方法,方便fn内部调用 //这里是实现子类中通过this._super()调用父类同名函数的关键 this._super = _super[name]; //此时再调用子类,子类里面的this._super已经指向了父类同名的函数,即_super[name] var ret = fn.apply(this, arguments); //this._super = tmp; //返回执行结果 return ret; }; })(name, prop[name]) : prop[name]; } //此“Class”与外头那个“Class”是两个东西 //这个Class实为子类的构造函数 function Class() { //不是A.extends({/* */})过程中 if ( !initializing && this.init ) this.init.apply(this, arguments); } //前面处理好的prototype绑定给子类的prototype Class.prototype = prototype; //修正构造函数 Class.prototype.constructor = Class; //赋予子类可被继续继承的功能 Class.extend = arguments.callee; return Class; }; })();
如果想更深入了解js继承,请继续往下查看文章
本文向大家介绍理解javascript对象继承,包括了理解javascript对象继承的使用技巧和注意事项,需要的朋友参考一下 先从一个问题进行研究深入,什么是javascript对象继承? 比如我们有一个“动物”对象的构造函数。 还有一个“猫”对象的构造函数。 我们知道猫也属于动物,如果这个猫对象想要继承动物对象的属性,我们该怎么做呢? 构造函数绑定 使用构造函数绑定是最简单的方法,使用call
本文向大家介绍JavaScript中的继承之类继承,包括了JavaScript中的继承之类继承的使用技巧和注意事项,需要的朋友参考一下 继承简介 在JS中继承是一个非常复杂的话题,比其他任何面向对象语言中的继承都复杂得多。在大多数其他面向对象语言中,继承一个类只需使用一个关键字即可。在JS中想要达到继承公用成员的目的,需要采取一系列措施。JS属于原型式继承,得益于这种灵活性,我们既可以
本文向大家介绍JavaScript中的对象继承关系,包括了JavaScript中的对象继承关系的使用技巧和注意事项,需要的朋友参考一下 我们今天就来看一下继承中的类继承以及类继承和原型继承的混用,所谓类继承,就是使用call或者apply方法来进行冒充继承: 像上面这种就是我们要使用的类继承,用这种继承,我们可以访问类中的方法和属性,但是无法访问父类原型中的方法和属性,这种方法别名冒充继承,顾
本文向大家介绍浅谈JavaScript对象与继承,包括了浅谈JavaScript对象与继承的使用技巧和注意事项,需要的朋友参考一下 JavaScript是我在C语言之后接触的第二门编程语言,大一暑假的时候在图书馆找了一本中国人写的JavaScript程序设计来看。那个时候在编程方面几乎还是小白,再加上那本书根本没有提JavaScript的编程机制,又有一些误导性的话,一直以来对JavaScript
本文向大家介绍JavaScript 类继承,包括了JavaScript 类继承的使用技巧和注意事项,需要的朋友参考一下 示例 继承的工作方式与其他面向对象的语言相同:在超类上定义的方法可在扩展子类中访问。 如果子类声明了自己的构造函数,则它必须先通过调用父构造函数,super()然后才能进行访问this。
面向对象编程很重要的一个方面,就是对象的继承。A 对象通过继承 B 对象,就能直接拥有 B 对象的所有属性和方法。这对于代码的复用是非常有用的。 大部分面向对象的编程语言,都是通过“类”(class)实现对象的继承。传统上,JavaScript 语言的继承不通过 class,而是通过“原型对象”(prototype)实现,本章介绍 JavaScript 的原型链继承。 ES6 引入了 class
本文向大家介绍浅谈Javascript中的对象和继承,包括了浅谈Javascript中的对象和继承的使用技巧和注意事项,需要的朋友参考一下 Javascript是一门函数式编程语言,Javascript当中函数是核心,在Javascript中函数也是对象,函数对象在创建的时候会被添加属性和方法。 在Javascript中函数对象有两种调用方式,一种是new关键字的调用,另一种是没有new关键字的调
问题内容: 我已经阅读了很多有关javascript中“继承”的文章。其中一些使用,而另一些则建议。我读得越多,我就越困惑,因为它似乎存在着无穷无尽的变体来解决继承问题。 有人可以告诉我最可接受的方式吗(如果有的话,是事实上的标准)吗? (我想要一个可以扩展或的基础对象。) 问题答案: 简单:并非在所有环境中均受支持,但可以填充。除此之外,两者具有不同的目的:只需创建一个对象从其他继承,同时 还