javascript设计模式之鸭子类型和多态
作者:闲人不梦卿
本文参考曾探编写的JavaScript设计模式与开发实践
设计模式的实现都遵循一条原则,即“找出程序中变化的地方,并将变化封装起来”。一个程序的设计总是可以分为可变的部分和不变的部分。当我们找出可变的部分,并且把这些部分封装起来,那么剩下的就是不变和稳定的部分。这些不变和稳定的部分是非常容易复用的。这也是设计模式为什么描写的是可复用面向对象软件基础的原因。
1.鸭子类型
鸭子类型的通俗说法是:“如果它走起路来像鸭子,叫起来也是鸭子,那么它就是鸭子。”
鸭子类型专业解释:例如,在不使用鸭子类型的语言中,我们可以编写一个函数,它接受一个类型为"鸭子"的对象,并调用它的"走"和"叫"方法,但它只能接受鸭子类型的对象,否则报错。在使用鸭子类型的语言中,这样的一个函数可以接受一个任意类型的对象,并调用它的"走"和"叫"方法。如果这些需要被调用的方法不存在,那么将引发一个运行时错误。任何拥有这样的正确的"走"和"叫"方法的对象都可被函数调用这种行为就是符合鸭子类型。
所以如果弱类型语言(js)函数需要接收参数,为保证健壮性,则应先判断参数类型,并判断参数是否包含需要访问的方法、属性。只有当这些条件满足时,程序才真正处理调用参数的方法、参数
var duck = { sing: function() { console.log('嘎嘎嘎'); } } var chicken = { sing: function() { console.log('嘎嘎嘎'); } } var choir = [] //合唱团 function joinChoir(duck) { if (duck && typeof duck.sing === 'function') { choir.push(duck) console.log('合唱队添加了一个成员'); } } joinChoir(duck) joinChoir(chicken) // 大合唱 for (let i = 0; i < choir.length; i++) { choir[i].sing() }
2.多态
2.1 java多态
对面向对象来说,多态分为编译时多态和运行时多态。其中编译时多态是静态的,主要是指方法的重载,它是根据参数列表的不同来区分不同的方法。通过编译之后会变成两个不同的方法,在运行时谈不上多态。而运行时多态是动态的,它是通过动态绑定来实现的,也就是大家通常所说的多态性。
在java里,多态是同一个行为具有不同表现形式或形态的能力,即对象多种表现形式的体现,就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。在简单来说,编译时对象是父类类型,到真正运行时,对象才可以知道具体是哪个子类类型,才知道调用哪个子类中实现的方法
Java 实现多态有 3 个必要条件:继承、重写和向上转型。只有满足这 3 个条件,开发人员才能够在同一个继承结构中使用统一的逻辑实现代码处理不同的对象,从而执行不同的行为。
- 继承:在多态中必须存在有继承关系的子类和父类。
- 重写:子类对父类中某些方法进行重新定义,在调用这些方法时就会调用子类的方法。
- 向上转型:在多态中需要将子类的引用赋给父类对象,只有这样该引用才既能可以调用父类的方法,又能调用子类的方法。
由此可以得出使用多态的好处,我们可以很好的完成代码的解耦和工作,加强代码的可扩展性,是代码更加灵活,在不改变原有接口方法的情况下简化流程等,总结一下就是:
- 减耦合
- 增强可以替换性
- 可扩展性
- 灵活性等…
举个生活例子,如下图所示:使用手机扫描二维码支付时,二维码并不知道客户是通过何种方式进行支付,只有通过二维码后才能判断是走哪种支付方式执行对应流程。
举个代码例子,创建 Figure 类,在该类中首先定义存储二维对象的尺寸,然后定义有两个参数的构造方法,最后添加 area() 方法,该方法计算对象的面积。代码如下:
public class Figure { double dim1; double dim2; Figure(double d1, double d2) { // 有参的构造方法 this.dim1 = d1; this.dim2 = d2; } double area() { // 用于计算对象的面积 System.out.println("父类中计算对象面积的方法,没有实际意义,需要在子类中重写。"); return 0; } }
创建继承自 Figure 类的 Rectangle 子类,该类调用父类的构造方法,并且重写父类中的 area() 方法。代码如下:
public class Figure { double dim1; double dim2; Figure(double d1, double d2) { // 有参的构造方法 this.dim1 = d1; this.dim2 = d2; } double area() { // 用于计算对象的面积 System.out.println("父类中计算对象面积的方法,没有实际意义,需要在子类中重写。"); return 0; } }
创建继承自 Figure 类的 Triangle 子类,该类与 Rectangle 相似。代码如下:
public class Rectangle extends Figure { Rectangle(double d1, double d2) { super(d1, d2); } double area() { System.out.println("长方形的面积:"); return super.dim1 * super.dim2; } }
创建 Test 测试类,在该类的 main() 方法中首先声明 Figure 类的变量 figure,然后分别为 figure 变量指定不同的对象,并调用这些对象的 area() 方法。代码如下:
public class Rectangle extends Figure { Rectangle(double d1, double d2) { super(d1, d2); } double area() { System.out.println("长方形的面积:"); return super.dim1 * super.dim2; } }
从上述代码可以发现,无论 figure 变量的对象是 Rectangle 还是 Triangle,它们都是 Figure 类的子类,因此可以向上转型为该类,从而实现多态。
2.2 js多态
多态的实际含义:是同一操作作用于不同的对象上面,可以产生不同的解释和不同的执行结果。换句话说,给不同的对象发送同一个消息的时候,这些对象会根据这个消息分别给出不同的反馈。
来举例说明一下多态的实际含义:
主人家里养了两只动物,分别是一只鸭和一只鸡,当主人向它们发出“叫”的命令时,鸭会“嘎嘎嘎”地叫,而鸡会“咯咯咯”地叫。这两只动物都会以自己的方式来发出叫声。它们同样“都是动物,并且可以发出叫声”,但根据主人的指令,它们会各自发出不同的叫声。
var Duck = function() { } Duck.prototype.sing = function() { console.log('嘎嘎嘎'); } var Chicken = function() { } Chicken.prototype.sing = function() { console.log('嘎嘎嘎'); } function sing(animal) { if (animal && (typeof animal.sing === 'function')) { animal.sing() } } sing(new Duck()) sing(new Chicken())
多态背后的思想:是将“做什么”和“谁去做以及怎样去做”分离开来,也就是将“不变的事物”与 “可能改变的事物”分离开来。在这个故事中,动物都会叫,这是不变的,但是不同类型的动物具体怎么叫是可变的。把不变的部分隔离出来,把可变的部分封装起来,这给予了我们扩展程序的能力,程序看起来是可生长的,也是符合开放—封闭原则的,相对于修改代码来说,仅仅增加代码就能完成同样的功能,这显然优雅和安全得多。
多态的实现:多态的思想实际上是把“做什么”和“谁去做”分离开来,要实现这一点,归根结底先要消除类型之间的耦合关系。如果类型之间的耦合关系没有被消除,那么我们在makeSound 方法中指定了发出叫声的对象是某个类型,它就不可能再被替换为另外一个类型。在 Java 中,可以通过向上转型来实现多态。而 JavaScript 的变量类型在运行期是可变的。一个JavaScript 对象,既可以表示 Duck 类型的对象,又可以表示 Chicken 类型的对象,这意味着 JavaScript 对象的多态性是与生俱来的。这种与生俱来的多态性并不难解释。JavaScript 作为一门动态类型语言,它在编译时没有类型检查的过程,既没有检查创建的对象类型,又没有检查传递的参数类型。
多态的最根本好处:你不必再向对象询问“你是什么类型”而后根据得到的答案调用对象的某个行为——你只管调用该行为就是了,其他的一切多态机制都会为你安排妥当。换句话说,多态最根本的作用就是通过把过程化的条件分支语句转化为对象的多态性,从而消除这些条件分支语句
多态的最根本好处,可以用下面这个例子很好地诠释:在电影的拍摄现场,当导演喊出“action”时,主角开始背台词,照明师负责打灯光,后面的群众演员假装中枪倒地,道具师往镜头里撒上雪花。在得到同一个消息时,每个对象都知道自己应该做什么。如果不利用对象的多态性,而是用面向过程的方式来编写这一段代码,那么相当于在电影开始拍摄之后,导演每次都要走到每个人的面前,确认它们的职业分工(类型),然后告诉他们要做什么。如果映射到程序中,那么程序中将充斥着条件分支语句。利用对象的多态性,导演在发布消息时,就不必考虑各个对象接到消息后应该做什么。对象应该做什么并不是临时决定的,而是已经事先约定和排练完毕的。每个对象应该做什么,已经成为了该对象的一个方法,被安装在对象的内部,每个对象负责它们自己的行为。所以这些对象可以根据同一个消息,有条不紊地分别进行各自的工作。
将行为分布在各个对象中,并让这些对象各自负责自己的行为,这正是面向对象设计的优点。
总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!