什么是双重分派

什么是分派(dispatch)

首先我们需要理解「分派」的含义。分派就是将方法调用与对应的具体方法绑定起来。而判断的依据有两点,这两者可称为「宗量」:

  1. 方法的接收者,也就是哪个对象调用了这个方法
  2. 方法的参数,调用方法时传递过来的参数

而不同的判断方法也就产生了不同的分派类型:

静态分派 V.S. 动态分派

单重分派 V.S. 多重分派

下面以我的理解简单地总结一下这几个名词的含义

静态分派

静态分派在编译期进行,只根据宗量的静态类型进行判断,在编译完成后不会发生改变。也可称为「编译期多态」

动态分派

动态分派在运行期进行,根据宗量的动态类型来判断需要调用哪个方法。也称「运行时多态」。

多重分派

虽然说 multiple 是「多重」的意思,但是实际上也可称为「双重分派」。

多重分派就是同时根据两种宗量的类型进行判断。单重分派就是只根据其中一种宗量进行判断。所以其实这两组词是从两个不同的角度来描述分派的类型,并不冲突。

Java 中的分派机制

在网上很多博客说 Java 中是单重分派;也有人说 Java 中实际上是静态双重分派,动态单重分派。下面是我的推论和总结。

Java 中的重载机制,是静态分派,其根据方法传入参数的静态类型进行判断,而方法的接收者,也直接在编译期就确定。所以是静态的双重分派。

而重写机制,肯定是动态分派了。但是其只根据方法接收者的动态类型进行判断。并不会根据方法参数的动态类型进行判断。所以是动态的单重分派。

那如果一个方法同时被重写和重载了呢,那这个方法属于什么分派机制呢?我认为,重写和重载本来就分别是编译期和运行期的多态机制,就是分开进行讨论的,在程序不同生命周期时间段的分派机制直接分开讨论并进行定义就好了。

访问者模式与伪双重分派

虽然说 Java 在编译期表现出了多重分派,但是也只能定性为单重分派语言,而这在设计代码时就会引起一些不便:在调用方法时无法根据参数的动态类型来分派对应的方法。所以我们只能曲线救国,可以通过以下两种方式实现伪动态双重分派。

1. instanceof

在方法内部可以通过 Java 的 instanceof 关键字来判断对象的动态类型。通过分支选择来进行对应的处理操作。但是这种思路会让处理代码变得十分臃肿,而且违背了「开闭原则」。每次要新增一个被处理类时,就要修改代码新增一个分支进行处理。具体的实现方法很简单,不赘述。

2. 访问者模式

设计模式中的「访问者模式」就是通过利用伪双重分派来实现的。主要思路如下:

设调用方为「类 A 」,被调用方为「类 B 」。通过反转调用方和被调用方,让类 A 重写方法来调用类 B 的重载方法。通过 this 变量将类 A 的动态类型传递给类 B 的重载方法,实现了伪双重分派。

在下面这个例子中,我们想让 Player.play() 方法根据 Instrument 的动态类型来分别调用对应的重载方法,从而进行不同的处理:

class Player {
void play(Instrument instrument) {
System.out.println("instrument notes");
} void play(Violin violin) {
System.out.println("player starts playing violin");
violin.note();
} void play(Piano piano) {
System.out.println("player starts playing piano");
piano.note();
}
} class Instrument {
void note() {
System.out.println("instrument notes");
} void accept(Player player) {
player.play(this);
}
} class Violin extends Instrument {
@Override
public void note() {
System.out.println("violin notes");
} @Override
public void accept(Player player) {
player.play(this);
}
} class Piano extends Instrument { @Override
public void note() {
System.out.println("piano notes");
} @Override
public void accept(Player player) {
player.play(this);
}
} public class Main { public static void main(String[] args) {
Player player = new Player();
System.out.println("-------双重分派-------");
Instrument[] instruments = {new Violin(), new Piano()};
for (Instrument instrument : instruments) {
instrument.accept(player);
} System.out.println("-------单重分派-------");
for (Instrument instrument : instruments) {
player.play(instrument);
}
}
}
// 结果
/*
-------双重分派-------
player starts playing violin
violin notes
player starts playing piano
piano notes
-------单重分派-------
instrument notes
instrument notes
*/

从上方的代码运行结果可以得知,在调用重载方法时,不会根据对象的动态类型进行判断,而是根据静态类型判断,只会调用 play(Instrument) 方法。当然,我们在重载 Player 的方法时可以直接调用 Instrument.note() 方法,让编译器去调用子类重写后的方法。

但是使用双重分派的意义在于, 把具体的处理逻辑放在调用者 Player 类的方法中,减少对 Instrument 的实现类的修改。事实上 Instrument 的子类只需要实现一个 accept(Player) 方法即可。

访问者模式的出现基本上也就是因为 Java 这类语言的单重分派特性吧。如果是多重分派的语言,就可以直接利用语言特性了。

参考文章:

  1. 面向对象语言的多分派、单分派、双重分派

  2. What is dispatching in JAVA?

多重分派(multiple dispatch)与访问者模式的更多相关文章

  1. 设计模式学习-使用go实现访问者模式

    访问者模式 定义 优点 缺点 适用范围 代码实现 什么是 Double Dispatch 参考 访问者模式 定义 访问者模式(Visitor):表示一个作用于某对象结构中的各元素的操作.它使你可以在不 ...

  2. C#设计模式总结 C#设计模式(22)——访问者模式(Vistor Pattern) C#设计模式总结 .NET Core launch.json 简介 利用Bootstrap Paginator插件和knockout.js完成分页功能 图片在线裁剪和图片上传总结 循序渐进学.Net Core Web Api开发系列【2】:利用Swagger调试WebApi

    C#设计模式总结 一. 设计原则 使用设计模式的根本原因是适应变化,提高代码复用率,使软件更具有可维护性和可扩展性.并且,在进行设计的时候,也需要遵循以下几个原则:单一职责原则.开放封闭原则.里氏代替 ...

  3. JAVA设计模式之访问者模式

    在阎宏博士的<JAVA与模式>一书中开头是这样描述访问者(Visitor)模式的: 访问者模式是对象的行为模式.访问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作需要 ...

  4. C#设计模式——访问者模式(Visitor Pattern)

    一.概述由于需求的改变,某些类常常需要增加新的功能,但由于种种原因这些类层次必须保持稳定,不允许开发人员随意修改.对此,访问者模式可以在不更改类层次结构的前提下透明的为各个类动态添加新的功能.二.访问 ...

  5. Java实现在访问者模式中使用反射

    集合类型在面向对象编程中很常用,这也带来一些代码相关的问题.比如,“怎么操作集合中不同类型的对象?” 一种做法就是遍历集合中的每个元素,然后根据它的类型而做具体的操作.这会很复杂,尤其当你不知道集合中 ...

  6. 设计模式《JAVA与模式》之访问者模式

    在阎宏博士的<JAVA与模式>一书中开头是这样描述访问者(Visitor)模式的: 访问者模式是对象的行为模式.访问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作需要 ...

  7. Java 设计模式系列(二三)访问者模式(Vistor)

    Java 设计模式系列(二三)访问者模式(Vistor) 访问者模式是对象的行为模式.访问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作需要修改的话,接受这个操作的数据结构则可以 ...

  8. Java设计模式—访问者模式

    原文地址:http://www.cnblogs.com/java-my-life/archive/2012/06/14/2545381.html 总结的太棒啦,导致自己看了都不想总结了...... 在 ...

  9. 《JAVA设计模式》之访问者模式(Visitor)

    在阎宏博士的<JAVA与模式>一书中开头是这样描述访问者(Visitor)模式的: 访问者模式是对象的行为模式.访问者模式的目的是封装一些施加于某种数据结构元素之上的操作.一旦这些操作需要 ...

随机推荐

  1. Kafka核心组件详解

    1.概述 对于Kafka的学习,在研究其系统模块时,有些核心组件是指的我们去了解.今天给大家来剖析一下Kafka的一些核心组件,让大家能够更好的理解Kafka的运作流程. 2.内容 Kafka系统设计 ...

  2. 浅谈Java-String到底是值传递还是引用传递?

    参数传递 Java 中的参数传递分为 "值传递""引用传递" 如果你学过 C/C++应该很好理解,就是所谓的 "值传递" 和 "指 ...

  3. Go语言学习——函数二 defer语句

    函数 package main import "fmt" // 函数:一段代码的封装 func f1(){ fmt.Println("Hello 中国!") } ...

  4. linux部署项目(前后端分离项目)

    参考博客 技术栈 路飞学城部署 vue + nginx + uwsgi + django + mysql + redis(就是一个key - value型数据库,缓存型数据库,内存型数据库) 部署步骤 ...

  5. 使用echo 无法正确清空文件存储大小

    在使用echo进行重定向文件的时候,会存在大小没有发生改变的现象 使用上面的方法遇到一个现象 ls -l 与 du -sh 得到的大小事是不同的 可以尝试下面的方面之后在进行对比 再看是否正确清除 使 ...

  6. 在 .NET 中使用 FixedTimeEquals 应对计时攻击

    计时攻击 在计算机安全中,计时攻击(Timing attack)是旁道攻击 (Side-channel attack) 的一种,而旁道攻击是根据计算机处理过程发出的信息进行分析,包括耗时,声音,功耗等 ...

  7. MySQL - 数据库设计步骤

    需求分析:分析用户的需求,包括数据.功能和性能需求. 概念结构设计:主要采用E-R模型进行设计,包括画E-R图. 逻辑结构设计:通过将E-R图转换成表,实现从E-R模型到关系模型的转换,进行关系规范化 ...

  8. element-ui table组件使用v-if时的问题

    element-ui项目中经常遇到需要使用v-if指令来根据情况动态显示隐藏某些列情况,这时就会出现滚动条样式异常.列错乱.列宽错乱等问题 解决办法:在el-table上添加:key="Ma ...

  9. 浅析DispatchProxy动态代理AOP

    浅析DispatchProxy动态代理AOP(代码源码) 最近学习了一段时间Java,了解到Java实现动态代理AOP主要分为两种方式JDK.CGLIB,我之前使用NET实现AOP切面编程,会用Fil ...

  10. SpringBoot + JWT + Redis 开源知识社区系统

    「Java学习+面试指南」一份涵盖大部分 Java 程序员所需要掌握的核心知识.准备 Java 面试,首选 JavaGuide!:https://javaguide.cn/ 你好,我是 Guide!这 ...