位置: 编程技术 - 正文

JavaScript 继承详解(四)(js中的继承)

编辑:rootadmin
Classical Inheritance in JavaScript。 Crockford是JavaScript开发社区最知名的权威,是JSON、JSLint、JSMin和ADSafe之父,是《JavaScript: The Good Parts》的作者。 现在是Yahoo的资深JavaScript架构师,参与YUI的设计开发。 这里有一篇文章详细介绍了Crockford的生平和著作。 当然Crockford也是我等小辈崇拜的对象。调用方式

首先让我们看下使用Crockford式继承的调用方式: 注意:代码中的method、inherits、uber都是自定义的对象,我们会在后面的代码分析中详解。

// 定义Person类 function Person(name) { this.name = name; } // 定义Person的原型方法 Person.method("getName", function() { return this.name; }); // 定义Employee类 function Employee(name, employeeID) { this.name = name; this.employeeID = employeeID; } // 指定Employee类从Person类继承 Employee.inherits(Person); // 定义Employee的原型方法 Employee.method("getEmployeeID", function() { return this.employeeID; }); Employee.method("getName", function() { // 注意,可以在子类中调用父类的原型方法 return "Employee name: " + this.uber("getName"); }); // 实例化子类 var zhang = new Employee("ZhangSan", ""); console.log(zhang.getName()); // "Employee name: ZhangSan"

这里面有几处不得不提的硬伤: 子类从父类继承的代码必须在子类和父类都定义好之后进行,并且必须在子类原型方法定义之前进行。 虽然子类方法体中可以调用父类的方法,但是子类的构造函数无法调用父类的构造函数。 代码的书写不够优雅,比如原型方法的定义以及调用父类的方法(不直观)。

当然Crockford的实现还支持子类中的方法调用带参数的父类方法,如下例子:

function Person(name) { this.name = name; } Person.method("getName", function(prefix) { return prefix + this.name; }); function Employee(name, employeeID) { this.name = name; this.employeeID = employeeID; } Employee.inherits(Person); Employee.method("getName", function() { // 注意,uber的第一个参数是要调用父类的函数名称,后面的参数都是此函数的参数 // 个人觉得这样方式不如这样调用来的直观:this.uber("Employee name: ") return this.uber("getName", "Employee name: "); }); var zhang = new Employee("ZhangSan", ""); console.log(zhang.getName()); // "Employee name: ZhangSan"

代码分析

首先method函数的定义就很简单了:

Function.prototype.method = function(name, func) { // this指向当前函数,也即是typeof(this) === "function" this.prototype[name] = func; return this; }; 要特别注意这里this的指向。当我们看到this时,不能仅仅关注于当前函数,而应该想到当前函数的调用方式。 比如这个例子中的method我们不会通过new的方式调用,所以method中的this指向的是当前函数。

inherits函数的定义有点复杂:

Function.method('inherits', function (parent) { // 关键是这一段:this.prototype = new parent(),这里实现了原型的引用 var d = {}, p = (this.prototype = new parent()); // 只为子类的原型增加uber方法,这里的Closure是为了在调用uber函数时知道当前类的父类的原型(也即是变量 - v) this.method('uber', function uber(name) { // 这里考虑到如果name是存在于Object.prototype中的函数名的情况 // 比如 "toString" in {} === true if (!(name in d)) { // 通过d[name]计数,不理解具体的含义 d[name] = 0; } var f, r, t = d[name], v = parent.prototype; if (t) { while (t) { v = v.constructor.prototype; t -= 1; } f = v[name]; } else { // 个人觉得这段代码有点繁琐,既然uber的含义就是父类的函数,那么f直接指向v[name]就可以了 f = p[name]; if (f == this[name]) { f = v[name]; } } d[name] += 1; // 执行父类中的函数name,但是函数中this指向当前对象 // 同时注意使用Array.prototype.slice.apply的方式对arguments进行截断(因为arguments不是标准的数组,没有slice方法) r = f.apply(this, Array.prototype.slice.apply(arguments, [1])); d[name] -= 1; return r; }); return this; }); 注意,在inherits函数中还有一个小小的BUG,那就是没有重定义constructor的指向,所以会发生如下的错误: var zhang = new Employee("ZhangSan", ""); console.log(zhang.getName()); // "Employee name: ZhangSan" console.log(zhang.constructor === Employee); // false console.log(zhang.constructor === Person); // true

改进建议

根据前面的分析,个人觉得method函数必要性不大,反而容易混淆视线。 而inherits方法可以做一些瘦身(因为Crockford可能考虑更多的情况,原文中介绍了好几种使用inherits的方式,而我们只关注其中的一种), 并修正了constructor的指向错误。

Function.prototype.inherits = function(parent) { this.prototype = new parent(); this.prototype.constructor = this; this.prototype.uber = function(name) { f = parent.prototype[name]; return f.apply(this, Array.prototype.slice.call(arguments, 1)); }; }; 调用方式: function Person(name) { this.name = name; } Person.prototype.getName = function(prefix) { return prefix + this.name; }; function Employee(name, employeeID) { this.name = name; this.employeeID = employeeID; } Employee.inherits(Person); Employee.prototype.getName = function() { return this.uber("getName", "Employee name: "); }; var zhang = new Employee("ZhangSan", ""); console.log(zhang.getName()); // "Employee name: ZhangSan" console.log(zhang.constructor === Employee); // true

有点意思

在文章的结尾,Crockford居然放出了这样的话:

I have been writing JavaScript for 8 years now, and I have never once found need to use an uber function. The super idea is fairly important in the classical pattern, but it appears to be unnecessary in the prototypal and functional patterns. I now see my early attempts to support the classical model in JavaScript as a mistake. 可见Crockford对在JavaScript中实现面向对象的编程不赞成,并且声称JavaScript应该按照原型和函数的模式(the prototypal and functional patterns)进行编程。 不过就我个人而言,在复杂的场景中如果有面向对象的机制会方便的多。 但谁有能担保呢,即使像jQuery UI这样的项目也没用到继承,而另一方面,像Extjs、Qooxdoo则极力倡导一种面向对象的JavaScript。 甚至Cappuccino项目还发明一种Objective-J语言来实践面向对象的JavaScript。

推荐整理分享JavaScript 继承详解(四)(js中的继承),希望有所帮助,仅作参考,欢迎阅读内容。

JavaScript 继承详解(四)(js中的继承)

文章相关热门搜索词:javascript继承原理,js继承如何实现,javascript继承原理,js继承的方法,javascript中继承,js中的继承,js继承的三种方式,js继承的三种方法,内容如对您有帮助,希望把文章链接给更多的朋友!

一个cssQuery对象 javascript脚本实现代码 /***@authorSupersha*@QQ:*/varcssQuery={//parent:用于存储当前节点的父节点的引用parent:document,select:function(selectorStr){varselectors=selectorStr.split("");//分隔字符串f

javascript Base类 包含基本的方法 scripttype="text/javascript"functionBase(){}//根抽象类Base.toBase=function(){//将一个对象转化成Base类的实例的方法returnnewBase();}Base.inherit=function(parent){//用于继承Base

浅析Javascript原型继承 推荐第1/2页 JS没有提供所谓的类继承,据说在2.0中要加入这种继承方式,但是要所有浏览器都实现2.0的特性那肯定又得N多年。昨天看了crockford的一个视频,里面讲

标签: js中的继承

本文链接地址:https://www.jiuchutong.com/biancheng/386481.html 转载请保留说明!

上一篇:JavaScript 继承详解(三)(js继承的方法)

下一篇:一个cssQuery对象 javascript脚本实现代码(在css中)

  • 税种有哪几种
  • 报个税时显示扣缴单位无有效的税费种认定信息
  • 增值税专用发票有效期是多长时间
  • 交通费中的高速费怎么算
  • 金税盘交钱
  • 银行本票与银行本票存款的区别
  • 无形资产价值评估收费
  • 现金流量表填报说明
  • 企业为清算状态是什么
  • 以前年度损益科目核算业务
  • 技术推广服务包含什么 奇瑞汽车
  • 没有收入的小规模纳税人可以无票入费用吗?
  • 股东出让个人股权流程
  • 停业的纳税人还交税吗
  • 一般纳税人临时工工资怎么入账
  • 退回产品怎么做分录
  • 没有进货票的商品可以销售吗
  • 付款金额比发票金额少怎么办
  • 汇兑损益在外币业务核算中有什么重要意义
  • 单位之间借款利息可以开票么
  • 税务局查账征收转为核定征收之程序
  • 买牛奶的公司会计怎么做账
  • 虚开发票可以做进项税额转出分录吗?
  • 劳务费增值税专用发票虚开一万元怎么补救
  • 银行汇票的背书转让金额
  • 专票丢了可以拿复印件在所得税前列支吗
  • 应交增值税是否影响企业利润
  • win10 bios设置
  • 华硕笔记本装win8
  • 鸿蒙系统怎么隐藏应用图标
  • 电脑屏幕字体模糊怎么办
  • php教程 ftp 函数
  • php基础入门教程
  • PHP:imagecolorclosestalpha()的用法_GD库图像处理函数
  • 补税的分录
  • 所得税分配表分录怎么写
  • 科技推广和应用服务业属于第几产业
  • 坏账准备收不回来怎么办
  • 论文笔记整理软件
  • 图像识别算法汇总表
  • 命令压缩文件
  • 认缴制情况下 公司一定要出资到位再注销吗
  • 工资可以当月计提当月发放怎么做账
  • mongorepository排序
  • 会计分录税金及附加有哪些
  • 工程物资月末必须结转吗
  • 交易性金融资产公允价值变动怎么算
  • 房产税征收标准2020
  • 冲销主营业务收入怎么做分录
  • 合适的库存产品包括
  • 没有入账的固定资产怎样做账
  • 资产负债表中各项目的期末余额
  • 银行承兑汇票记载事项
  • 税控盘会计分录怎么做
  • 进项税额转出怎么操作
  • 工业企业销售商品
  • mysql5.7慢查询
  • centos5.7
  • u盘装win7系统步骤
  • Win10预览版更新弹窗如何关闭
  • win8.1技巧
  • linux shell 数字转字符串
  • 如何删除win7
  • win10周年版
  • linux 命令连接
  • win7 c盘打不开
  • exec方法
  • unity learn premium
  • html:xt
  • 基于flask框架
  • 在html中写入javascript
  • javascript实现3D切换焦点图
  • JavaScript中的math.pi
  • cocos2dx4.0入门
  • 表单验证js
  • 外埠企业如何在经营地缴税
  • 新一轮税制改革的意义
  • 发票怎样上传?
  • 所得税的核定征收方法
  • 税务总局纪律作风
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设