时间:2021-05-18
由于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;//实例化父类,并把父类的实例方法及属性给prototypevar 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绑定给子类的prototypeClass.prototype = prototype;//修正构造函数Class.prototype.constructor = Class;//赋予子类可被继续继承的功能Class.extend = arguments.callee;return Class;};})();如果想更深入了解js继承,请继续往下查看文章
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
本文实例讲述了javascript继承机制。分享给大家供大家参考。具体分析如下:初学javascript一般很难理解Javascript语言的继承机制它没有"子
Javascript是唯一一个被广泛运用的原型式继承的语言,所以理解两种继承方式的差异是需要时间的。第一个主要差异就是Javascript使用原型链来继承:fu
学完了Javascript类和对象的创建之后,现在总结一下Javascript继承机制的实现。Javascript并不像Java那样对继承机制有严格明确的定义,
最近做web项目,接触了jquery等框架,虽然使用方便,但是还是想学习下Javascript,今天分享下最近对js原型继承的理解,不足之处欢迎指正。一、构造器
本文实例讲述了Javascript面向对象程序设计继承用法。分享给大家供大家参考,具体如下:1.关于继承:百度百科对继承的解释是:继承是指一个对象直接使用另一对