La signification réelle du polymorphisme est la suivante : la même opération appliquée à différents objets peut produire différentes interprétations et différents résultats d'exécution. En d’autres termes, lorsque le même message est envoyé à différents objets, ces objets donneront des retours différents en fonction du message.
Il n'est pas facile de comprendre le polymorphisme littéralement. Prenons exemple ci-dessous.
Le propriétaire a deux animaux à la maison, un canard et un poulet. Lorsque le propriétaire leur donne l'ordre "coin", le canard cancan et le poulet crient. Les deux animaux émettent des sons à leur manière. Ce sont également « tous des animaux et peuvent émettre des sons », mais selon les instructions du propriétaire, ils émettront chacun des sons différents.
En fait, il contient l'idée de polymorphisme. Ci-dessous, nous le présenterons en détail à travers le code.
1. Un morceau de code JavaScript "polymorphe"
Nous implémentons l'histoire ci-dessus en utilisant le code JavaScript comme suit :
var makeSound = function( animal ){ if ( animal instanceof Duck ){ console.log( '嘎嘎嘎' ); }else if ( animal instanceof Chicken ){ console.log( '咯咯咯' ); } }; var Duck = function(){}; var Chicken = function(){}; makeSound( new Duck() ); //嘎嘎嘎 makeSound( new Chicken() ); //咯咯咯
Ce code incarne en effet le "polymorphisme". Lorsque nous envoyons un message "d'appel" respectivement aux canards et aux poules, ils réagissent différemment en fonction de ce message. Mais un tel "polymorphisme" n'est pas satisfaisant. Si un animal est ajouté plus tard, comme un chien, évidemment l'aboiement du chien est "woof wouf wouf". . Modifier du code est toujours dangereux. Plus vous modifiez d'endroits, plus le risque d'erreurs de programme est grand, et lorsqu'il y a de plus en plus de types d'animaux, makeSound peut devenir une fonction énorme.
L'idée derrière le polymorphisme est de séparer « quoi » de « qui le fait et comment le faire », c'est-à-dire de séparer « les choses qui ne changent pas » des « choses qui peuvent changer ». Dans cette histoire, tous les animaux aboient, ce qui est constant, mais les noms spécifiques des différents types d'animaux sont variables. Isoler les parties constantes et encapsuler les parties variables nous donne la possibilité d'étendre le programme. Le programme semble extensible et conforme au principe ouvert-fermé. Par rapport à la modification du code, il ajoute seulement que le code peut accomplir la même fonction, ce qui est le cas. est évidemment beaucoup plus élégant et plus sûr.
2. Polymorphisme des objets
Ce qui suit est le code réécrit. Tout d'abord, nous isolons la partie inchangée, c'est-à-dire que tous les animaux émettent des sons :
var makeSound = function( animal ){ animal.sound(); };
Encapsulez ensuite les parties variables séparément. Le polymorphisme dont nous venons de parler fait en fait référence au polymorphisme de l'objet :
var Duck = function(){} Duck.prototype.sound = function(){ console.log( '嘎嘎嘎' ); }; var Chicken = function(){} Chicken.prototype.sound = function(){ console.log( '咯咯咯' ); }; makeSound( new Duck() ); //嘎嘎嘎 makeSound( new Chicken() ); //咯咯咯
Maintenant, nous envoyons le message « d'appel » aux canards et aux poules, et ils réagissent différemment après avoir reçu le message. Si un jour un autre chien est ajouté au monde animal, vous pouvez simplement ajouter du code à ce moment-là sans changer la fonction makeSound précédente, comme indiqué ci-dessous :
var Dog = function(){} Dog.prototype.sound = function(){ console.log( '汪汪汪' ); }; makeSound( new Dog() ); //汪汪汪
3. Vérification de type et polymorphisme
La vérification de type est un sujet incontournable avant de montrer le polymorphisme d'un objet, mais JavaScript est un langage typé dynamiquement qui ne nécessite pas de vérification de type. Afin de vraiment comprendre le but du polymorphisme, nous devons franchir un cap. langage tapé.
Les langages typés statiques effectuent des vérifications de correspondance de type au moment de la compilation. En prenant Java comme exemple, en raison d'une vérification de type stricte lors de la compilation du code, les variables ne peuvent pas se voir attribuer des valeurs de types différents. Cette vérification de type rend parfois le code rigide. Le code est le suivant :
.String str; str = abc; //没有问题 str = 2; //报错
Essayons maintenant de changer l'exemple ci-dessus de faire cancaner des canards et des poulets en code Java :
public class Duck { //鸭子类 public void makeSound(){ System.out.println( 嘎嘎嘎 ); } } public class Chicken { //鸡类 public void makeSound(){ System.out.println( 咯咯咯 ); } } public class AnimalSound { public void makeSound( Duck duck ){ //(1) duck.makeSound(); } } public class Test { public static void main( String args[] ){ AnimalSound animalSound = new AnimalSound(); Duck duck = new Duck(); animalSound.makeSound( duck ); //输出:嘎嘎嘎 } }
Nous avons réussi à faire cancaner les canards, mais si nous voulons que les poules cancanent maintenant, nous trouvons que c'est impossible. Parce que la méthode makeSound de la classe AnimalSound en (1) est stipulée par nos soins pour n'accepter que les paramètres de type Duck :
public class Test { public static void main( String args[] ){ AnimalSound animalSound = new AnimalSound(); Chicken chicken = new Chicken(); animalSound.makeSound( chicken ); //报错,只能接受Duck类型的参数 } }
某些时候,在享受静态语言类型检查带来的安全性的同时,我们亦会感觉被束缚住了手脚。
为了解决这一问题,静态类型的面向对象语言通常被设计为可以向上转型:当给一个类变量赋值时,这个变量的类型既可以使用这个类本身,也可以使用这个类的超类。这就像我们在描述天上的一只麻雀或者一只喜鹊时,通常说“一只麻雀在飞”或者“一只喜鹊在飞”。但如果想忽略它们的具体类型,那么也可以说”一只鸟在飞“。
同理,当Duck对象和Chicken对象的类型都被隐藏在超类型Animal身后,Duck对象和Chicken对象就能被交换使用,这是让对象表现出多态性的必经之路,而多态性的表现正是实现众多设计模式的目标。
4. 使用继承得到多态效果
使用继承来得到多态效果,是让对象表现出多态性的最常用手段。继承通常包括实现继承和接口继承。本节我们讨论实现继承,接口继承的例子请参见第21章。
我们先创建一个Animal抽象类,再分别让Duck和Chicken都继承自Animal抽象类,下述代码中(1)处和(2)处的赋值语句显然是成立的,因为鸭子和鸡也是动物:
public abstract class Animal { abstract void makeSound(); //抽象方法 } public class Chicken extends Animal{ public void makeSound(){ System.out.println( 咯咯咯 ); } } public class Duck extends Animal{ public void makeSound(){ System.out.println( 嘎嘎嘎 ); } } Animal duck = new Duck(); //(1) Animal chicken = new Chicken(); //(2)
现在剩下的就是让AnimalSound类的makeSound方法接受Animal类型的参数,而不是具体的Duck类型或者Chicken类型:
public class AnimalSound{ public void makeSound( Animal animal ){ //接受Animal类型的参数 animal.makeSound(); } } public class Test { public static void main( String args[] ){ AnimalSound animalSound= new AnimalSound (); Animal duck = new Duck(); Animal chicken = new Chicken(); animalSound.makeSound( duck ); //输出嘎嘎嘎 animalSound.makeSound( chicken ); //输出咯咯咯 } }
5. JavaScript的多态
从前面的讲解我们得知,多态的思想实际上是把“做什么”和“谁去做”分离开来,要实现这一点,归根结底先要消除类型之间的耦合关系。如果类型之间的耦合关系没有被消除,那么我们在makeSound方法中指定了发出叫声的对象是某个类型,它就不可能再被替换为另外一个类型。在Java中,可以通过向上转型来实现多态。
而JavaScript的变量类型在运行期是可变的。一个JavaScript对象,既可以表示Duck类型的对象,又可以表示Chicken类型的对象,这意味着JavaScript对象的多态性是与生俱来的。
这种与生俱来的多态性并不难解释。JavaScript作为一门动态类型语言,它在编译时没有类型检查的过程,既没有检查创建的对象类型,又没有检查传递的参数类型。在2节的代码示例中,我们既可以往makeSound函数里传递duck对象当作参数,也可以传递chicken对象当作参数。
由此可见,某一种动物能否发出叫声,只取决于它有没有makeSound方法,而不取决于它是否是某种类型的对象,这里不存在任何程度上的“类型耦合”。这正是我们从上一节的鸭子类型中领悟的道理。在JavaScript中,并不需要诸如向上转型之类的技术来取得多态的效果。
6. 多态在面向对象程序设计中的作用
有许多人认为,多态是面向对象编程语言中最重要的技术。但我们目前还很难看出这一点,毕竟大部分人都不关心鸡是怎么叫的,也不想知道鸭是怎么叫的。让鸡和鸭在同一个消息之下发出不同的叫声,这跟程序员有什么关系呢?
Martin Fowler在《重构:改善既有代码的设计》里写到:
多态的最根本好处在于,你不必再向对象询问“你是什么类型”而后根据得到的答案调用对象的某个行为——你只管调用该行为就是了,其他的一切多态机制都会为你安排妥当。
换句话说,多态最根本的作用就是通过把过程化的条件分支语句转化为对象的多态性,从而消除这些条件分支语句。
Martin Fowler的话可以用下面这个例子很好地诠释:
在电影的拍摄现场,当导演喊出“action”时,主角开始背台词,照明师负责打灯光,后面的群众演员假装中枪倒地,道具师往镜头里撒上雪花。在得到同一个消息时,每个对象都知道自己应该做什么。如果不利用对象的多态性,而是用面向过程的方式来编写这一段代码,那么相当于在电影开始拍摄之后,导演每次都要走到每个人的面前,确认它们的职业分工(类型),然后告诉他们要做什么。如果映射到程序中,那么程序中将充斥着条件分支语句。
利用对象的多态性,导演在发布消息时,就不必考虑各个对象接到消息后应该做什么。对象应该做什么并不是临时决定的,而是已经事先约定和排练完毕的。每个对象应该做什么,已经成为了该对象的一个方法,被安装在对象的内部,每个对象负责它们自己的行为。所以这些对象可以根据同一个消息,有条不紊地分别进行各自的工作。
将行为分布在各个对象中,并让这些对象各自负责自己的行为,这正是面向对象设计的优点。
再看一个现实开发中遇到的例子,这个例子的思想和动物叫声的故事非常相似。
假设我们要编写一个地图应用,现在有两家可选的地图API提供商供我们接入自己的应用。目前我们选择的是谷歌地图,谷歌地图的API中提供了show方法,负责在页面上展示整个地图。示例代码如下:
var googleMap = { show: function(){ console.log( '开始渲染google地图' ); } }; var renderMap = function(){ googleMap.show(); }; renderMap(); // 输出: 开始渲染google地图
后来因为某些原因,要把谷歌地图换成百度地图,为了让renderMap函数保持一定的弹性,我们用一些条件分支来让renderMap函数同时支持谷歌地图和百度地图:
var googleMap = { show: function(){ console.log( '开始渲染google地图' ); } }; var baiduMap = { show: function(){ console.log( '开始渲染baidu地图' ); } }; var renderMap = function( type ){ if ( type === 'google' ){ googleMap.show(); }else if ( type === 'baidu' ){ baiduMap.show(); } }; renderMap( 'google' ); // 输出: 开始渲染google地图 renderMap( 'baidu' ); // 输出: 开始渲染baidu地图
可以看到,虽然renderMap函数目前保持了一定的弹性,但这种弹性是很脆弱的,一旦需要替换成搜搜地图,那无疑必须得改动renderMap函数,继续往里面堆砌条件分支语句。
我们还是先把程序中相同的部分抽象出来,那就是显示某个地图:
var renderMap = function( map ){ if ( map.show instanceof Function ){ map.show(); } }; renderMap( googleMap ); // 输出: 开始渲染google地图 renderMap( baiduMap ); // 输出: 开始渲染baidu地图
现在来找找这段代码中的多态性。当我们向谷歌地图对象和百度地图对象分别发出“展示地图”的消息时,会分别调用它们的show方法,就会产生各自不同的执行结果。对象的多态性提示我们,“做什么”和“怎么去做”是可以分开的,即使以后增加了搜搜地图,renderMap函数仍然不需要做任何改变,如下所示:
var sosoMap = { show: function(){ console.log( '开始渲染soso地图' ); } }; renderMap( sosoMap ); // 输出: 开始渲染soso地图
在这个例子中,我们假设每个地图API提供展示地图的方法名都是show,在实际开发中也许不会如此顺利,这时候可以借助适配器模式来解决问题。
以上就是本文的全部内容,很全面,以生动的举例来帮助大家学习多态,希望大家能够真正的有所收获。