`
simohayha
  • 浏览: 1403341 次
  • 性别: Icon_minigender_1
  • 来自: 火星
社区版块
存档分类
最新评论

javascript中的继承

阅读更多
首先在js中的继承是Prototype-based 。不像一般的oo语言,比如java,c++是Class-based 的。

我们来比较一下这两种方式 :

1 在Class-based 中的类和实例是不同的实体,而在Prototype-based中所有对象都是实例。

2 在Class-based中定义一个类使用class关键字来定义。实例化一个对象,使用构造方法。而在Prototype-based,定义和创建一个对象都使用构造器函数。

3 在Class-based中构造一个继承,需要定义一个类,然后作为存在的类的子类。而在Prototype-based中,则是需要标记这个对象作为构造器函数的prototype 。

4 在Class-based中继承属性通过class chain。而在Prototype-based则是通过prototype chain。

下来我们来看一个我们需要实现的继承体系(后面有图):

Manager 和WorkerBee 继承Employee ,而SalesPerson 和Engineer继承WorkerBee.

其中,Employee有一个name 属性,Manager有一个reports 属性,SalesPerson 有一个quota 属性,Engineer有一个machine 属性。

function Employee () {
this.name = "";
this.dept = "general";
} 

function Manager () {
this.reports = [];
}
Manager.prototype = new Employee; 
function WorkerBee () {
this.projects = [];
}
WorkerBee.prototype = new Employee; 
function SalesPerson () {
   this.dept = "sales";
   this.quota = 100;
}
SalesPerson.prototype = new WorkerBee; 
function Engineer () {
   this.dept = "engineering";
   this.machine = "";
}
Engineer.prototype = new WorkerBee; 


这只是一个非常非常简单的例子,不过这里我们可以看到在javascript中的继承,就是通过构造函数的prototype来进行构造的。



下面又有一个问题,那就是想给一个对象在运行时加入属性怎么办,或许你想到该这么做:

mark.bonus = 3000


可是这样做的话只是mark 对象拥有了一个bonus属性,而不是WorkerBee 拥有这个属性。

如果你想要加一个属性被这个构造函数的所有对象所共享,那么你就必须这么做:

Employee.prototype.specialty = "none"; 


这里的话,就是给整个继承体系加入了一个specialty属性。

现在假设我们有下面的代码(这里已经给各个构造器加入了一些参数:

jane = new Engineer("belau"); 


然后jane的属性的值如下:

引用
jane.name == "";
jane.dept == "general";
jane.projects == [];
jane.machine == "belau


可以发现这时我们无法指定一个继承属性的值,不过没关系,我们可以这样修改我们的程序:

function Employee (name, dept) {
this.name = name || "";
this.dept = dept || "general";
}

function Engineer (name, projs, mach) {
this.base = WorkerBee;
this.base(name, "engineering", projs);
this.machine = mach || "";
} 

function WorkerBee (name,dept,projs) {
this.base=Employee;
this.base(name,dept);
this.projects = projs || [];
}
WorkerBee.prototype = new Employee; 



现在我们可以分析下
jane = new Engineer("Doe, Jane", ["navigator", "javascript"], "belau");
调用这句时,都发生了什么:

1  当js看见new操作符,它创建一个新的普通对象,并且设置它的__proto__ 属性为Engineer.prototype。

2 new 操作符传递这个新的对象作为Engineer 构造器的this的值。

其实最主要做的事就是上面的两件,剩下的都是很简单的函数调用.



我们来看一下在继承中,假设你要存取一个对象的一个属性时会发生什么:

1 首先检测这个值是否在本地存在。

2 如果不存在本地变量,那么就检测prototype chain (使用__proto__ )。

3 如果属性在prototype chain中存在那么就返回这个值,否则这个对象就不存在这个值。

比如下面的例子:

function Employee () {
   this.dept = "general";
}
Employee.prototype.name = ""; 

function WorkerBee () {
this.projects = [];
}
WorkerBee.prototype = new Employee; 

amy = new WorkerBee; 

Employee.prototype.name = "Unknown"; 


可以看到打印出Unknown.

通过上面我们知道当你使用new操作符创建一个对象的时候,js会设置这个对象的__proto__作为构造器函数的prototype ,因此我们能够通过这个来测试prototype chain。

假设我们创建了一个chris 对象:
chris = new Engineer("Pigman, Chris", ["jsd"], "fiji");


那么下面的语句都会返回true:

chris.__proto__ == Engineer.prototype;
chris.__proto__.__proto__ == WorkerBee.prototype;
chris.__proto__.__proto__.__proto__ == Employee.prototype;
chris.__proto__.__proto__.__proto__.__proto__ == Object.prototype;
chris.__proto__.__proto__.__proto__.__proto__.__proto__ == null;


最后要说的是,在js中并没有提供多重继承.

不过我们可以这样模拟实现:


function Hobbyist (hobby) {
   this.hobby = hobby || "scuba";
} 

function Engineer (name, projs, mach, hobby) {
   this.base1 = WorkerBee;
   this.base1(name, "engineering", projs);
   this.base2 = Hobbyist;
   this.base2(hobby);
   this.machine = mach || "";
}
Engineer.prototype = new WorkerBee; 

dennis = new Engineer("Doe, Dennis", ["collabra"], "hugo") 


这样做不是很好,这是因为每次new的时候都会分配Hobbyist的空间。

可是如果我们给Hobbyist添加一个属性的话,dennis 对象将不能继承这个新的属性.





  • 描述: A simple object hierarchy
  • 大小: 225.6 KB
16
2
分享到:
评论
3 楼 simohayha 2008-02-28  
恩,那里描述不清,我改一下...
2 楼 笨笨狗 2008-02-28  
所以,并不像楼主所说:
“最后设置这个对象的__proto__ 为WorkerBee.prototype”
这样的话无法解释在构造器中调用原型链方法,比如:
function Test(){
    this.init();//此时如果还未设置__proto__,必然出错
}
Test.prototype.init = function(){}
1 楼 笨笨狗 2008-02-28  
new一个function的时候
var obj = new Test(argsList)
可以这么理解:
1、首先,创建一个空对象tmp
tmp = {}
2、然后,将这个tmp的__proto__指向Test.prototype。不过__proto__貌似只有ff支持?其他浏览器内部应该也是差不多,反正就是要让这个对象能引用构造函数的原型方法
tmp.__proto__ = Test.prototype
3、最后,将tmp作为context执行构造函数,也就是
Test.apply(tmp,argsList)

相关推荐

    JavaScript中继承原理与用法实例入门

    JavaScript中的继承是面向对象编程的重要概念,它允许一个对象(子对象)继承另一个对象(父对象)的属性和方法。在JavaScript中,由于其动态类型和基于原型的特性,实现继承的方式比传统的面向对象语言更为灵活,但...

    JavaScript中继承用法实例分析

    在给定的文件内容中,我们看到一个具体的JavaScript继承的例子,主要分为以下几个步骤: 1. 定义父类`Person`,这个类有一个原型方法`walk`和`sayHello`,它们分别用于定义行走和打招呼的行为。 ```javascript ...

    JavaScript中继承的一些示例方法与属性参考

    JavaScript继承是面向对象编程(OOP)中的一个核心概念,它允许一个对象(称为子类或派生类)继承另一个对象(称为父类或基类)的属性和方法。JavaScript是一种基于原型的编程语言,它实现继承的方式与其他基于类的...

    base-object:JavaScript 中继承的实现

    JavaScript 中继承的实现 怎么建? git clone 这个仓库 切换到新目录 npm install npm run build 用法 var Person = BaseObject . extend ( { init : function ( name , age ) { this . name = name ; this . ...

    JavaScript继承

    JavaScript中的继承是一个核心概念,它在面向对象编程中扮演着至关重要的角色。与其他面向对象语言相比,JavaScript的继承机制更为复杂。在Java或C++等语言中,继承通常只需要一个关键字,如`extends`,但在...

    JavaScript中实现继承的八种方式

    详细介绍了JavaScript中继承的实现方式以及其优缺点分析。

    JavaScript实现继承的几种方式

    寄生组合继承解决了组合继承中父类构造函数会被调用两次的问题,通过创建父类的一个不被使用的实例,然后将子类的原型指向这个实例的原型。 ```javascript function inheritPrototype(subType, superType) { let ...

    JavaScript学习之三 — JavaScript实现继承的7种方式

    在JavaScript中,选择合适的继承方式取决于具体需求。理解并熟练掌握这些继承方式,能让你在编写代码时更加游刃有余。文章中的`inheritance.html`和`inheritance.js`文件可能包含示例代码,帮助你直观地了解每种继承...

    javascript控件开发之继承关系

    在这个主题中,“javascript控件开发之继承关系”主要探讨的是如何利用JavaScript的面向对象特性来构建和组织控件的层次结构,以及如何通过继承来实现代码的复用和模块化。 在JavaScript中,继承是基于原型...

    JavaScript中的类继承

    JavaScript中的类继承是一种模拟传统面向对象编程中类概念的方式,因为JavaScript本身是一种基于原型的面向对象语言。在JavaScript中,对象可以直接从其他对象继承属性和方法,而不是通过类的实例化。这种继承机制...

    实现JavaScript中继承的三种方式

    在原型链继承中,子类型的原型对象指向父类型的实例,从而使得子类型的实例可以继承父类型原型上的属性和方法。原型链继承允许子类型访问父类型的实例属性和方法,这在JavaScript中是通过设置子类型的prototype指向...

    学习javascript面向对象 javascript实现继承的方式

    本文将详细介绍六种实现JavaScript继承的方法。 1. 原型链继承 原型链继承是通过改变子类型的原型对象来实现继承的。子类型构造函数的原型被替换为超类型构造函数的一个实例,这样子类型就可以继承超类型的所有属性...

    javascript继承之为什么要继承.docx

    JavaScript 中的继承机制是指子类继承父类的属性和方法,使得子类可以拥有父类的所有特征。继承是面向对象编程的基本机制之一,它可以实现代码复用、提高编程效率和增强代码的可维护性。 在 JavaScript 中,继承是...

    理解Javascript原型继承原理

    在JavaScript中,原型继承是一种非常核心且独特的机制,它使得对象能够继承其他对象的属性和方法。本文旨在深入探讨这一机制,并通过具体的示例代码帮助读者更好地理解其中的概念。 #### 二、基本概念 在JavaScript...

    JavaScript继承机制研究.pdf

    在本文中,我们将深入探讨JavaScript继承机制的实现方式,并对基于原型的继承、构造函数方式继承、组合继承、寄生式继承等继承机制进行了总结归纳和分析。 基于原型的继承 JavaScript是一门基于原型的语言,它不像...

Global site tag (gtag.js) - Google Analytics