多态(Polymorphism)按字面的意思就是“多种状态”,同样的行为(方法)在不同对象上有不同的状态。
在OOP中很多地方都要用到多态的特性,比如同样是点击鼠标右键,点击快捷方式、点击桌面空白处、点击任务栏等弹出的菜单都是不同的。
方法重写(override):
即子类定义一个与父类名字相同的方法,以此覆盖父类方法,以此来实现不同的功能。
1 function Animal(){} 2 var AnimalP = Animal.prototype; 3 AnimalP.eat = function(food){ 4 console.log('这个动物正在吃' + food); 5 }; 6 7 function Snake(){} 8 var SnakeP = extend(Snake,Animal);//extend函数请看上一节 9 /*snake没有对eat方法重写,继承的父类eat()方法*/ 10 function Dog(){} 11 var DogP = extend(Dog,Animal); 12 DogP.eat = function(food){ 13 /*对eat()方法重写*/ 14 /*上一章讲过,也可以在这里通过 Animal.eat.call(this,food)调用父方法;*/ 15 console.log("这只狗正在吃"+food); 16 }; 17 18 function Cat(){} 19 var CatP = extend(Cat,Animal); 20 CatP.eat = function(food){ 21 console.log("这只猫正在吃"+food); 22 }; 23 var snake = new Snake(); 24 snake.eat('老鼠');//log:这个动物正在吃老鼠 25 var dog = new Dog(); 26 dog.eat('骨头');//log:这只狗正在吃骨头 27 var cat = new Cat(); 28 cat.eat('鱼');//log:这只猫正在吃鱼
抽象类(abstract class):
上面的代码中,Snake类没有实现自己的eat()方法,但有的时候我们希望子类一定要有某个方法(抽象方法),这样可以规范子类的行为,这时候就要用到抽象类,
ES5、ES6都没有抽象类的概念的,所以我们只能通过模拟来实现,让我们接着上面的代码,假如我们要把Animal的eat()方法定义为抽象方法:
1 AnimalP.eat = function(food){ 2 /*定义抽象方法(虚函数),如果子类没有重写这个方法,在执行这方法的时候就会抛出错误*/ 3 throw '"' + this.constructor.name + "'类没有eat()方法"; 4 }; 5 function Snake(){} 6 var SnakeP = extend(Snake,Animal); 7 var snake = new Snake(); 8 snake.eat('老鼠');//throw:"Snake'类没有eat()方法
方法重载(overload):
我们一定写过这样的函数,根据传入的参数不一样(类型、参数个数),方法的运行结果也不一样:
1 var run = function(speed){ 2 if(typeof speed == 'number'){ 3 console.log('跑的速度有' + speed + 'm/s'); 4 }else if(typeof speed == 'string'){ 5 console.log('跑的速度有' + speed); 6 } 7 } 8 run(15);//log:跑的速度有15m/s 9 run('20KM/h');//log:跑的速度有20KM/h
但上面这样写明显代码难维护,可以把run方法作为一个接口,根据参数的类型执行不同方法,用在类中就向下面一样:
1 function Dog(){} 2 var DogP = Dog.prototype; 3 DogP.run = function(speed){ 4 if(typeof speed == 'number'){ 5 this._runNumber(speed); 6 }else if(typeof speed == 'string'){ 7 this._runString(speed); 8 }else{ 9 throw '参数不匹配'; 10 } 11 } 12 DogP._runString = function(speed){ 13 console.log('这只狗跑的速度有' + speed); 14 } 15 DogP._runNumber = function(speed){ 16 console.log('这只狗跑的速度有' + speed + 'm/s'); 17 } 18 var dog = new Dog(); 19 dog.run(15);//log:这只狗跑的速度有15m/s 20 dog.run('20KM/h');//log:这只狗跑的速度有20KM/h 21 dog.run([]);//throw:参数不匹配
这就是方法重载的模拟,但实际上,ES5、ES6、typescipt都不支持语法上的方法重载,typescipt也只是支持函数重载。
这是多态的另一种实现方式。
Demo by ES6:
1 class Animal{ 2 eat(food){ 3 throw '"' + this.constructor.name + "'类没有eat()方法"; 4 } 5 } 6 class Snake extends Animal{} 7 class Dog extends Animal{ 8 eat(food){ 9 console.log("这只狗正在吃"+food); 10 } 11 } 12 class Cat extends Animal{ 13 eat(food){ 14 console.log("这只猫正在吃"+food); 15 } 16 } 17 let snake = new Snake(); 18 snake.eat('老鼠');//throw:"Snake'类没有eat()方法 19 let dog = new Dog(); 20 dog.eat('骨头');//log:这只狗正在吃骨头 21 let cat = new Cat(); 22 cat.eat('鱼');//log:这只猫正在吃鱼
Demo by TypeScript:
1 abstract class Animal{//定义抽象类Animal 2 constructor(){} 3 abstract eat(food: string){} 4 /*定义抽象方法eat(),并且限定传入的参数类型是string, 5 还可以定义返回值,接口等,如果子类不符合限定的规范,编译的时候就会报错。 6 */ 7 } 8 class Snake extends Animal{}//报错,无法通过编译,因为没有定义eat()抽象方法 9 class Dog extends Animal{ 10 eat(food: string){ 11 console.log("这只狗正在吃"+food); 12 } 13 } 14 class Cat extends Animal{ 15 eat(food: string){ 16 console.log("这只猫正在吃"+food); 17 } 18 } 19 let dog = new Dog(); 20 dog.eat('骨头');//log:这只狗正在吃骨头 21 let cat = new Cat(); 22 cat.eat('鱼');//log:这只猫正在吃鱼
后话
如果你喜欢作者的文章,记得收藏,你的点赞是对作者最大的鼓励;
面向对象的主要知识点在这里就讲完了,这些东西仅仅是基础,我讲的肯定不够完善,仅仅是为了让大家快速入门,建议大家有时间的话还是系统的看书学习一下js OOP;
本系列还有最后一章,会把前几张讲的知识点通过一个案例整合在一起,让大家可以更好的消化吸收,大概会酝酿两周的时间;
大家有什么疑问可以留言或私信作者,作者尽量第一时间回复大家;
如果老司机们觉得那里可以有不恰当的,或可以表达的更好的,欢迎指出来,我会尽快修正、完善。