TDD测试驱动的javascript开发(3) ------ javascript的继承

简介: 说起面向对象,人们就会想到继承,常见的继承分为2种:接口继承和实现继承。接口继承只继承方法签名,实现继承则继承实际的方法。 由于函数没有签名,在ECMAScript中无法实现接口继承,只支持实现继承。 1. 原型链 1.1 原型链将作为实现继承的主要方法,基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。 构造函数---原型---实例 之间的关系: 每一个构造函数都

说起面向对象,人们就会想到继承,常见的继承分为2种:接口继承和实现继承。接口继承只继承方法签名,实现继承则继承实际的方法。

由于函数没有签名,在ECMAScript中无法实现接口继承,只支持实现继承。

1. 原型链

1.1 原型链将作为实现继承的主要方法,基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法。

构造函数---原型---实例 之间的关系:

每一个构造函数都有一个原型对象,原型对象包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针。

function SuperType() {
	this.property = true;
}

SuperType.prototype.getSuperValue = function() {
	return this.property;
};

function SubType() {
	this.subproperty = false;
}

SubType.prototype = new SuperType();    //通过原型链实现继承

SubType.prototype.getSubValue = function() {
	return this.subproperty;
};

var subInstance = new SubType();

var superInstance = new SuperType();

TestCase("test extends",{
	
	"test superInstance property should be true" : function() {
		assertEquals(true,superInstance.property);
	},
	"test superInstance getSuperValue() should be return true" : function() {
		assertEquals(true,superInstance.getSuperValue());
	},
	"test subInstance property should be false" : function() {
		assertEquals(false,subInstance.subproperty);
	},
	"test subInstance could visit super method " : function() {
		assertEquals(true,subInstance.getSuperValue());    //SubType继承SuperType,并调用父类的方法
	}
});

注:要区分开父类和子类的属性名称,否则子类的属性将会覆盖父类的同名属性值:看如下代码:


function SubType() {
	this.property = false;
}
SubType.prototype.getSubValue = function() {
	return this.property;
};
function SuperType() {
	this.property = true;
}

SuperType.prototype.getSuperValue = function() {
	return this.property;
};

SubType.prototype = new SuperType();    //通过原型链实现继承


var subInstance = new SubType();

var superInstance = new SuperType();

TestCase("test extends",{
	
	"test superInstance property should be true" : function() {
		assertEquals(true,superInstance.property);    //父类的property值为true
	},
	"test superInstance getSuperValue() should be return true" : function() {
		assertEquals(true,superInstance.getSuperValue());   //superInstance调用方法
	},
	"test subInstance property should be false" : function() {
		assertEquals(false,subInstance.property);     //子类的property属值为false
	},
	"test subInstance could visit super method " : function() {
		assertEquals(false,subInstance.getSuperValue());    //SubType继承SuperType,并调用父类的方法,可以属性被覆盖了,返回false
	}
});
续:当然,如果我们不要求对属性值进行初始化的时候,就不必考虑这个问题,我们会采用上一章讲的构造函数模式+原型模式来创建类和实现继承关系。

当以读取模式访问一个实例属性时,首先会在实例中搜索该属性,如果没有找到该属性,则继续在实例的原型中寻找。在通过原型链实现继承的情况下,会继续沿着原型链继续向上

subExtends.getSuperValue()
首先在实例中查找,然后在SubType.prototype,最后在SuperType.prototype中找到。

补充: 所有函数的默认原型都是Object的实例,因此默认原型都会包含一个内部指针,指向Object.prototype,这也正是所有自定义类型都会继承toString()、valueOf()等默认方法的原因。

1.2 确定原型和实例的关系

 方法一:使用instanceof 操作符     ----     只要该实例是原型链中出现过的构造函数,结果就会返回true

function SuperType() {
	this.property = true;
}

SuperType.prototype.getSuperValue = function() {
	return this.property;
};

function SubType() {
	this.subproperty = false;
}

SubType.prototype = new SuperType();    //通过原型链实现继承

SubType.prototype.getSubValue = function() {
	return this.subproperty;
};

var subInstance = new SubType();

TestCase("test extends",{
	
	"test subInstance should instanceof Object" : function() {
		assertInstanceOf(Object,subInstance);
	},
	"test subInstance should instanceof SuperType " : function() {
		assertInstanceOf(SuperType,subInstance);
	},
	"test subInstance should instanceof SubType " : function() {
		assertInstanceOf(SubType,subInstance);
	}
});

 方法二:使用isPrototypeOf()方法     ----    只要原型链中出现过该原型,都可以说是该原型链所派生的实例的原型,结果会返回true

function SuperType() {
	this.property = true;
}

SuperType.prototype.getSuperValue = function() {
	return this.property;
};

function SubType() {
	this.subproperty = false;
}

SubType.prototype = new SuperType();    //通过原型链实现继承

SubType.prototype.getSubValue = function() {
	return this.subproperty;
};

var subInstance = new SubType();

TestCase("test extends",{
	
	"test subInstance isPrototypeOf  Object" : function() {
		assertEquals(true,Object.prototype.isPrototypeOf(subInstance));
	},
	"test subInstance isPrototypeOf SuperType " : function() {
		assertEquals(true,SuperType.prototype.isPrototypeOf(subInstance));
	},
	"test subInstance isPrototypeOf SubType " : function() {
		assertEquals(true,SubType.prototype.isPrototypeOf(subInstance));
	}
});

注:在实践中,我们很少会单独的使用原型链,因为它存在两个问题:

一、引用类型值的原型:包含引用类型值的原型属性会被所有实例共享,这也正是为什么要在构造函数中定义属性,而不在原型中定义属性的原因

二、创建子类型的实例时,不能向超类型的构造函数中传递参数。

1.3  借用构造函数(伪造对象  ----  经典继承)

原理: 在子类型构造函数的内部调用超类型的构造函数

function SuperType(name) {
	this.name = name;
	this.friends = ['tong','feng'];
}

function SubType(name,age) {
	SuperType.call(this,name);
}

var subInstance = new SubType("tongtong",26);

subInstance.friends.push('ty');

var subInstance2 = new SubType("fengfeng",27);

TestCase("test constructor extends",{
	"test subInstance friends property" : function() {
		assertEquals("ty",subInstance.friends[2]);    //将ty  push到数组
	},
	"test subInstance2 friends length" : function() {
		assertEquals(2,subInstance2.friends.length);   //subInstance2 friends属性没有改变
	}
});

1.4 组合继承(经典伪继承)  -----   推荐模式

将原型链和借用构造函数的技术组合到一块,使用原型链实现对原型属性和方法的继承,通过借用构造函数来实现对实例属性的继承

function SuperType(name) {
	this.name = name;
	this.friends = ['tong','feng'];
}
SuperType.prototype.sayName = function() {
	return this.name;
};

function SubType(name,age) {
	//继承属性
	SuperType.call(this,name);                   //调用构造函数
	this.age = age;
}
//继承方法
SubType.prototype = new SuperType();                //调用构造函数

SubType.prototype.sayAge = function() {
	return this.age;
};

var subInstance = new SubType("tongtong",26);

subInstance.friends.push('ty');

var subInstance2 = new SubType("fengfeng",27);

TestCase("test extends",{
	
	"test subInstance name property" : function() {
		assertEquals("tongtong",subInstance.sayName());
	},
	"test subInstance2 name property" : function() {
		assertEquals("fengfeng",subInstance2.sayName());
	},
	"test subInstance friends property" : function() {
		assertEquals("ty",subInstance.friends[2]);    //将ty  push到数组
	},
	"test subInstance2 friends length" : function() {
		assertEquals(2,subInstance2.friends.length);   //subInstance2 friends属性没有改变
	}
});

这种继承的缺点:将会2次调用构造函数,性能一般,解决办法:参见1.7寄生组合式继承

1.5 原型式继承

这种方法没有严格意义上的构造函数,思想是借助原型可以基于已有的对象创建新的对象,同时还不必因此创建自定义类型。

它要求你必须有一个对象可以作为另一个对象的基础。如果有这么一个对象,可以把它传递给object()函数,然后该函数就会返回一个新对象。

function object(o) {
	function F() {
	}

	F.prototype = o;
	return new F();
}

var person = {
	name : "tongtong",
	friends : [ "feng", "tong" ]
};

var another = object(person);
another.name = "newtong";
another.friends.push('lan');

var other = object(person);

TestCase("test prototype extends",{
	
	"test another is extends  person name property" : function() {
		assertEquals("newtong",another.name);
	},
	"test person firends property length is 3" : function() {
		assertEquals(3,person.friends.length);
	},
	"test other firends property length is 3" : function() {
		assertEquals(3,other.friends.length);
	}
});

person 作为另一个对象的基础,我们把它传入到object()中,然后该函数就会返回一个新对象,这个对象将person做为原型。

ECMAScript通过Object.create()方法规范了原型式继承,在传入一个参数的时候,Object.create()与object()方法的行为相同。

var person = {
	name : "tongtong",
	friends : [ "feng", "tong" ]
};
var another = Object.create(person);
another.name = "lisa";
another.friends.push("lan");
TestCase("test prototype extends",{
	
	"test another is extends  person name property" : function() {
		assertEquals("lisa",another.name);
	},
	"test person firends property length is 3" : function() {
		assertEquals(3,person.friends.length);
	}
});

如果传入2个参数,第二个参数会覆盖同名参数

var person = {
	name : "tongtong",
	friends : [ "feng", "tong" ]
};
var another = Object.create(person, {
	name : {
		value : "claire"
	}
});

another.friends.push('lalala');
TestCase("test prototype extends",{
	
	"test another is extends  person name property" : function() {
		assertEquals("claire",another.name);
	},
	"test person firends property length is 3" : function() {
		assertEquals(3,person.friends.length);
	}
});

在只想让一个对象与另一个对象保持类似的情况下,又没必要创建构造函数的时候,原型式继承OK.(它和原型模式一样哦,引用类型的值都会被共享)。

1.6寄生式继承

思路:与寄生构造和工厂模式类似,创建一个仅用于封装继承过程的函数,该函数在内部以某种方式来增强对象,最后再像它真的做了所以工作一样返回。

//这是一个函数哦
function createAnother(original) {
//	var clone = object(original);   //创建一个新对象
	var clone = Object.create(original);
	clone.sayThis = function() {        //增强对象
		return original; 
	};
	return clone;                   //返回对象
};

var person = {
		name : "tong",
		friends : ['tong','feng']
};

var another = createAnother(person);

TestCase("test prototype extends",{
	"test another is extends  person name property" : function() {
		assertEquals("tong",another.sayThis().name);
	}
});
;
缺点:使用寄生式继承来为对象添加函数,会因为函数不能复用而降低效率(和构造函数模式相似)

1.7寄生组合式继承  ------  最佳方案

所谓寄生组合式继承,即通过借用构造函数来继承属性,通过原型链的混合形式来继承方法,

思路:不必为了指定子类型的原型而调用超类型的构造函数,我们所需的无非就是超类型原型的一个副本而已。

本质上:使用寄生式继承来继承超类型的原型,然后再将结果指定给子类型的原型

特点:寄生式继承是引用类型最理想的继承方式

function SuperType(name) {
	this.name = name;
	this.colors = ['red','green'];
}

SuperType.prototype.sayName = function() {
	return this.name;
};
/**
 * 1.创建父类型的一个副本
 * 2.弥补创建副本时所丢失的constructor属性
 * 3.将新的副本赋值给子类型的原型
 * @param subType
 * @param superType
 */
function inheritPrototype(subType,superType) {
	var object = Object.create(superType.prototype);//创建对象
	object.constructor = subType;         //增强对象
	subType.prototype = object;			//指定对象
}

function SubType(name,age) {
	SuperType.call(this,name);   //调用构造函数
	this.age = age;
}

inheritPrototype(SubType,SuperType);

SubType.prototype.sayAge = function() {
	return this.age;
};



var subInstance1 = new SubType('feng',28); 
subInstance1.colors.push('yellow');

var subInstance2 = new SubType('tong',25);

TestCase("test parasitic extends",{
	"test subInstance1 name property should be feng" : function() {
		assertEquals("feng",subInstance1.name);    
	},
	"test subInstance2 name property should be tong" : function() {
		assertEquals('tong',subInstance2.name);   
	},
	"test subInstance1 colors property length should be 3" : function() {
		assertEquals(3,subInstance1.colors.length);   
	},
	"test subInstance2 colors property length should be 2" : function() {
		assertEquals(2,subInstance2.colors.length);   
	},
	"test subInstance1 sayAge method should be return 28" : function() {
		assertEquals(28,subInstance1.sayAge());   
	},
	"test subInstance2 sayAge method should be return 25" : function() {
		assertEquals(25,subInstance2.sayAge());   
	},
	"test subInstance1 should be instanceof SuperType" : function() {
		assertInstanceOf(SuperType,subInstance1);   
	}
	
});


1.8总结

1.8.1 ECMAScript 支持面向对象编程,但不使用类或者接口,对象可以在代码执行过程中创建和增强。

1.8.2 创建对象的几种方式:

构造函数模式:可以创建自定义的引用类型,使用new操作符

     缺点:在每个实例上都要重新创建,无法复用,包括函数

     优点:与对象的松耦合

     适用场合:当属性或者方法不做共享属性或者方法的时候(比如引用类型的属性),不建议单独使用。

原型模式:使用构造函数的prototype属性来指定那些共享的属性和方法

     优点:所有成员都可以共享属性和方法

     缺点:没有私有属性值

     适用场合:所以的属性和方法都可以被共享的时候,不建议单独使用

组合使用构造函数和原型模式:使用构造函数定义实例属性,使用原型定义共享属性和方法。

     优点:解决了原型模式共享引用类型的属性的问题,也解决了构造函数不能共享属性的问题。

     缺点:实现继承的时候,将会2次调用父类型的构造函数。性能问题。

     使用场合:使用最广泛的定义引用类型的一种默认模式。

动态原型模式:保持了构造函数和原型的优点,把所有的信息封装在构造函数内,在有必要的情况下进行初始化原型。

稳妥构造函数模式:适合在安全环境中使用。

1.8.3 javascript的继承:

javascript主要通过原型链事实现继承。

原型链的继承:原型链的构建是通过将一个类型的实例赋值给另一个构造函数的原型实现的。

     缺点:对象实例共享所有继承的属性和方法,因此不宜单独使用。

借用构造函数继承:在子类构造函数的内部调用超类型的构造函数。call()       apply()  

     缺点:没有函数的复用,不建议单独使用

组合继承:使用原型链继承共享的属性和方法,而通过借用构造函数继承实例属性。

     缺点:将会调用两次构造函数,会有性能问题

原型式继承:可以在不必预定义构造函数的情况下实现继承,其本质是执行对给定对象的浅复制,而复制的副本还可以得到进一步的改造。

     优点:在没有必要创建构造函数,只是让一个对象与另一个对象保持类似的情况下,原型式继承OK.

     缺点:共享属性和方法

寄生式继承:与原型式继承非常相似,基于对象获某些信息创建一个对象,然后增强对象,最后返回对象

     优点:解决组合继承多次调用父类的构造函数而导致低效率问题。(可以与组合模式一起使用)

     缺点:使用寄生式继承来为对象添加函数,会由于不能做到函数复用而降低效率,与构造函数模式类似。

寄生组合式继承:集寄生式继承和组合继承的优点于一身,是实现基于类型继承的最有效方式。





目录
相关文章
|
13天前
|
IDE 测试技术 开发工具
10个必备Python调试技巧:从pdb到单元测试的开发效率提升指南
在Python开发中,调试是提升效率的关键技能。本文总结了10个实用的调试方法,涵盖内置调试器pdb、breakpoint()函数、断言机制、logging模块、列表推导式优化、IPython调试、警告机制、IDE调试工具、inspect模块和单元测试框架的应用。通过这些技巧,开发者可以更高效地定位和解决问题,提高代码质量。
110 8
10个必备Python调试技巧:从pdb到单元测试的开发效率提升指南
|
10天前
|
数据采集 人工智能 自然语言处理
Midscene.js:AI 驱动的 UI 自动化测试框架,支持自然语言交互,生成可视化报告
Midscene.js 是一款基于 AI 技术的 UI 自动化测试框架,通过自然语言交互简化测试流程,支持动作执行、数据查询和页面断言,提供可视化报告,适用于多种应用场景。
109 1
Midscene.js:AI 驱动的 UI 自动化测试框架,支持自然语言交互,生成可视化报告
|
1月前
|
Web App开发 JavaScript 前端开发
Node.js开发
Node.js开发
47 13
|
2月前
|
存储 JavaScript 前端开发
深入浅出Node.js后端开发
在数字化时代的浪潮中,后端开发作为连接用户与数据的桥梁,扮演着至关重要的角色。本文将以Node.js为例,深入探讨其背后的哲学思想、核心特性以及在实际项目中的应用,旨在为读者揭示Node.js如何优雅地处理高并发请求,并通过实践案例加深理解。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的视角和思考。
|
2月前
|
Web App开发 开发框架 JavaScript
深入浅出Node.js后端开发
本文将带你领略Node.js的魅力,从基础概念到实践应用,一步步深入理解并掌握Node.js在后端开发中的运用。我们将通过实例学习如何搭建一个基本的Web服务,探讨Node.js的事件驱动和非阻塞I/O模型,以及如何利用其强大的生态系统进行高效的后端开发。无论你是前端开发者还是后端新手,这篇文章都会为你打开一扇通往全栈开发的大门。
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
智能化软件测试:AI驱动的自动化测试策略与实践####
本文深入探讨了人工智能(AI)在软件测试领域的创新应用,通过分析AI技术如何优化测试流程、提升测试效率及质量,阐述了智能化软件测试的核心价值。文章首先概述了传统软件测试面临的挑战,随后详细介绍了AI驱动的自动化测试工具与框架,包括自然语言处理(NLP)、机器学习(ML)算法在缺陷预测、测试用例生成及自动化回归测试中的应用实例。最后,文章展望了智能化软件测试的未来发展趋势,强调了持续学习与适应能力对于保持测试策略有效性的重要性。 ####
|
2月前
|
Web App开发 开发框架 JavaScript
深入浅出Node.js后端开发
在这篇文章中,我们将一起探索Node.js的奇妙世界。无论你是刚接触后端开发的新手,还是希望深化理解的老手,这篇文章都适合你。我们将从基础概念开始,逐步深入到实际应用,最后通过一个代码示例来巩固所学知识。让我们一起开启这段旅程吧!
|
1月前
|
Web App开发 JavaScript 前端开发
深入浅出Node.js后端开发
本文将带领读者从零基础开始,一步步深入到Node.js后端开发的精髓。我们将通过通俗易懂的语言和实际代码示例,探索Node.js的强大功能及其在现代Web开发中的应用。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的见解和技巧,让你的后端开发技能更上一层楼。
|
2月前
|
JavaScript 前端开发 API
深入理解Node.js事件循环及其在后端开发中的应用
本文旨在揭示Node.js的核心特性之一——事件循环,并探讨其对后端开发实践的深远影响。通过剖析事件循环的工作原理和关键组件,我们不仅能够更好地理解Node.js的非阻塞I/O模型,还能学会如何优化我们的后端应用以提高性能和响应能力。文章将结合实例分析事件循环在处理大量并发请求时的优势,以及如何避免常见的编程陷阱,从而为读者提供从理论到实践的全面指导。
|
2月前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
65 1