0

0

Java中的多态分派机制(Static vs Dynamic)_静态分派与方法重载

P粉602998670

P粉602998670

发布时间:2026-02-26 13:30:11

|

190人浏览过

|

来源于php中文网

原创

静态分派发生在编译期,依据引用类型选择重载方法;动态分派发生在运行时,依据实际对象类型选择重写方法;二者协同工作,先静态后动态。

java中的多态分派机制(static vs dynamic)_静态分派与方法重载

静态分派发生在编译期,只看引用类型

Java 的方法重载(overload)是典型的静态分派场景。编译器在编译时就决定调用哪个重载版本,依据是变量的**声明类型**(即引用类型),而不是运行时实际对象类型。

常见错误现象:NullPointerException 没抛出,但调用的却是 print(Object) 而不是你预期的 print(String),因为传入的是 Object obj = new String("hello") —— 编译器只认 objObject 类型。

  • 重载方法选择完全由参数的**编译期类型**决定,和运行时值无关
  • final 修饰的引用不会改变静态分派结果;var 推导出的类型也参与静态分派
  • 泛型擦除后,List<string></string>List<integer></integer> 在重载中都视为 List,容易导致意外匹配

动态分派依赖运行时对象类型,只对实例方法生效

方法重写(override)触发动态分派,JVM 在运行时根据对象的实际类(而非引用类型)查找并执行对应版本的 invokevirtual 指令。

使用场景:多态接口调用、模板方法模式、策略切换等需要运行时行为变化的地方。

立即学习Java免费学习笔记(深入)”;

企奶奶
企奶奶

一款专注于企业信息查询的智能大模型,企奶奶查企业,像聊天一样简单。

下载
  • 只有 public / protected / 包级可见的非 static、非 final、非 private 实例方法才参与动态分派
  • static 方法看似能“重写”,实则是隐藏(hiding),调用目标由引用类型决定 —— 属于静态分派
  • private 方法不能被重写,子类里同名方法是全新方法,不参与任何分派

重载 + 重写混用时,先静态再动态

当一个调用既涉及重载又涉及重写(比如父类引用指向子类对象,且多个重载方法中某个被重写),JVM 分两步走:先按参数类型选重载版本(编译期),再按对象实际类型选重写实现(运行期)。

容易踩的坑:show(List)show(ArrayList) 都存在时,传 new ArrayList<string>()</string> 会进 show(ArrayList);但如果把参数改成 Object list = new ArrayList(),就只能匹配 show(List) 或更宽泛的 show(Object) —— 因为静态分派阶段已丢失 ArrayList 类型信息。

  • 重载决策永远优先于重写,二者不在同一层机制里
  • 数组类型如 String[]Object[] 在重载中是不同签名,但 String[]Object 的子类型,可能触发意外交叉匹配
  • 自动装箱/拆箱会引入隐式重载候选,例如 int 可匹配 method(int)method(long)method(Integer),编译器按优先级选,但容易误判

如何验证当前调用走的是静态还是动态分派

最直接的方式是看字节码:javap -c YourClass。静态分派的方法调用通常对应 invokestaticstatic 方法)或 invokestatic/invokespecial(构造器、私有方法、super. 调用);而动态分派一定是 invokevirtual(实例方法)或 invokeinterface(接口方法)。

另一个轻量方式:在各重载/重写方法里加日志,用不同引用类型调用,观察输出是否随声明类型或实际类型变化。

  • 如果换引用类型(如 A a = new B()B b = new B())导致调用方法不同,说明是静态分派
  • 如果只换实际对象类型(如 A a = new B()A a = new C()),调用方法随之变化,说明是动态分派
  • 注意 IDE 的“Find Usages”可能误导——它基于源码分析,不反映运行时绑定逻辑
静态分派和动态分派的边界其实很窄,但一旦混淆,bug 往往藏在类型转换、泛型、自动装箱这些看似无害的操作里。真正难的不是理解概念,而是读代码时能下意识区分“这里编译器看到什么”和“JVM 运行时拿到什么”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
python中print函数的用法
python中print函数的用法

python中print函数的语法是“print(value1, value2, ..., sep=' ', end=' ', file=sys.stdout, flush=False)”。本专题为大家提供print相关的文章、下载、课程内容,供大家免费下载体验。

192

2023.09.27

python print用法与作用
python print用法与作用

本专题整合了python print的用法、作用、函数功能相关内容,阅读专题下面的文章了解更多详细教程。

14

2026.02.03

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

27

2025.11.27

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

27

2025.11.27

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

27

2025.11.27

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

588

2024.08.29

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

1

2026.02.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.9万人学习

C# 教程
C# 教程

共94课时 | 10.2万人学习

Java 教程
Java 教程

共578课时 | 72.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号