对javascript继承的理解


Posted in Javascript onOctober 11, 2016

由于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 相关文章推荐
关于跨站脚本攻击问题
Dec 22 Javascript
基于JavaScript实现继承机制之构造函数+原型链混合方式的使用详解
May 07 Javascript
JavaScript加强之自定义event事件
Sep 21 Javascript
jquery操作cookie插件分享
Jan 14 Javascript
完美兼容各大浏览器获取HTTP_REFERER方法总结
Jun 24 Javascript
jq实现左滑显示删除按钮,点击删除实现删除数据功能(推荐)
Aug 23 Javascript
js基本算法:冒泡排序,二分查找的简单实例
Oct 08 Javascript
Bootstrap基本模板的使用和理解1
Dec 14 Javascript
JS数组操作中的经典算法实例讲解
Jul 26 Javascript
Vue resource三种请求格式和万能测试地址
Sep 26 Javascript
Node.js 实现抢票小工具 & 短信通知提醒功能
Oct 22 Javascript
vuex中遇到的坑,vuex数据改变,组件中页面不渲染操作
Nov 16 Javascript
Javascript动画效果(3)
Oct 11 #Javascript
JavaScript实现自动切换图片代码
Oct 11 #Javascript
Javascript动画效果(2)
Oct 11 #Javascript
Javascript动画效果(1)
Oct 11 #Javascript
原生Javascript和jQuery做轮播图简单例子
Oct 11 #Javascript
jQuery progressbar通过Ajax请求实现后台进度实时功能
Oct 11 #Javascript
javascript之with的使用(阿里云、淘宝使用代码分析)
Oct 11 #Javascript
You might like
php使用类继承解决代码重复的问题
2015/02/11 PHP
PHP树形结构tree类用法示例
2019/02/01 PHP
javascript的函数、创建对象、封装、属性和方法、继承
2011/03/10 Javascript
jquery $(this).attr $(this).val方法使用介绍
2013/10/08 Javascript
JS实现根据出生年月计算年龄
2014/01/10 Javascript
Bootstrap实现input控件失去焦点时验证
2016/08/04 Javascript
AngularJS验证信息框架的封装插件用法【w5cValidator扩展插件】
2016/11/03 Javascript
JS实现获取来自百度,Google,soso,sogou关键词的方法
2016/12/21 Javascript
angularJs使用$watch和$filter过滤器制作搜索筛选实例
2017/06/01 Javascript
vue写h5页面的方法总结
2019/02/12 Javascript
Vue学习之常用指令实例详解
2020/01/06 Javascript
python生成随机验证码(中文验证码)示例
2014/04/03 Python
Python基于动态规划算法计算单词距离
2015/07/25 Python
Python编程实现两个文件夹里文件的对比功能示例【包含内容的对比】
2017/06/20 Python
基于python socketserver框架全面解析
2017/09/21 Python
Python构建网页爬虫原理分析
2017/12/19 Python
python快速建立超简单的web服务器的实现方法
2018/02/17 Python
Python Django给admin添加Action的方法实例详解
2019/04/29 Python
python移位运算的实现
2019/07/15 Python
Python使用ffmpy将amr格式的音频转化为mp3格式的例子
2019/08/08 Python
Python 70行代码实现简单算式计算器解析
2019/08/30 Python
Python字典dict常用方法函数实例
2020/11/09 Python
雅诗兰黛美国官网:Estee Lauder美国
2016/07/21 全球购物
蒂娜商店:Tiina the Store
2019/12/07 全球购物
编程实现去掉XML的重复结点
2014/05/28 面试题
文秘专业毕业生就业推荐信
2013/11/08 职场文书
大三预备党员入党思想汇报
2014/01/08 职场文书
工作鉴定评语
2014/05/04 职场文书
安全伴我行演讲稿
2014/09/04 职场文书
大学拉赞助协议书范文
2014/09/26 职场文书
六五普法宣传标语
2014/10/06 职场文书
学生会宣传部竞选稿
2015/11/21 职场文书
2016个人廉洁自律承诺书
2016/03/25 职场文书
详解nodejs内置模块
2021/05/06 NodeJs
Python读写yaml文件
2022/03/20 Python
MySQL数据库实验实现简单数据库应用系统设计
2022/06/21 MySQL