什么是双重分派

什么是分派(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. MySQL实时在线备份恢复方案

    开源Linux 长按二维码加关注~ 上一篇:2020年MySQL数据库面试题总结 快照和复制技术的结合可以保证我们得到一个实时的在线MySQL备份解决方案. 当主库发生误操作时,只需要恢复备库上的快照 ...

  2. gogin web框架部署学习

    首先去git上面找了一个gin框架拿来学习gin web开发: flipped-aurora/gin-vue-admin: 基于vite+vue3+gin搭建的开发基础平台(已完成setup语法糖版本 ...

  3. 升级gradle后。需要修改jenkin 编译java版本从1.8 到11

    错误提示 * What went wrong: A problem occurred evaluating project ':App'. > Failed to apply plugin 'c ...

  4. 精华!一张图进阶 RocketMQ

    前 言 大家好,我是三此君,一个在自我救赎之路上的非典型程序员. "一张图"系列旨在通过"一张图"系统性的解析一个板块的知识点: 三此君向来不喜欢零零散散的知识 ...

  5. Python | 内置函数(BIF)

    Python内置函数 | V3.9.1 | 共计155个 还没学完, 还没记录完, 不知道自己能不能坚持记录下去 1.ArithmeticError 2.AssertionError 3.Attrib ...

  6. 5分钟了解二叉树之AVL树

    转载请注明出处:https://www.cnblogs.com/morningli/p/16033733.html AVL树是带有平衡条件的二叉查找树,其每个节点的左子树和右子树的高度最多相差1.为了 ...

  7. 使用python获取交换机syslog日志并使用jQuery在html上展示

    需求 现网有部分pop点独立于海外,无法发送日志给内网日志服务器,同时最近网内有比较重要割接,所以临时写一个脚本来展示网内日志 思路 使用socket接收syslog数据,udp 514,数据部分格式 ...

  8. 关于我学git这档子事(3)

    对于如下报错: hint: Updates were rejected because a pushed branch tip is behind its remote hint: counterpa ...

  9. 『忘了再学』Shell基础 — 24、Shell正则表达式的使用

    目录 1.正则表达式说明 2.基础正则表达式 3.练习 (1)准备工作 (2)*练习 (3).练习 (4)^和$练习 (5)[]练习 (6)[^]练习 (7)\{n\}练习 (8)\{n,\}练习 ( ...

  10. django框架10

    内容概要 ajax结合sweetalert forms组件钩子函数 forms组件字段参数 forms组件字段类型 forms组件源码分析 cookie与session简介 django操作cooki ...