写作背景
基于JDK1.8,新版本变化概不负责。
问题描述
当一个类有多个重载方法,它们之间参数个数相同、存在相互转换关系时,在调用方法时,实际调用的哪个方法?
结论
先说结论,方法的重载选择是静态分派,即编译期确认调用的是哪个方法,这个选择的依据按照如下顺序优先级由高到低:
- 选择类型准确方法,找不到往下看
- 若参数为基本类型,按照byte > short > int > long >float >double 或char > int > long >float > double的顺序选择方法入参为基本类型的方法,找不到往下看
- 若参数为基本类型,选择包装类方法,找不到往下看
- 按照继承/实现层级从少到多的顺序选择入参为父类的方法,找不到往下看,若存在相同层级的父类入参方法编译无法通过
- 选择Object入参类型方法,找不到往下看
- 选择类型准确可变参数方法,找不到往下看;若参数为基本类型或基本类型包装类,选择基本类型准确或包装类可变参数方法,找不到往下看,同时存在基本类型和基本类型包装类可变参数方法编译无法通过
- 若参数为基本类型或包装类,按照byte > short > int > long >float >double 或char > int > long >float > double的顺序选择方法入参为基本类型可变参数方法,找不到往下看
- 选择Object可变参数方法找不到编译报错
静态类型
上面说到静态分派,啥意思呢,就是依据静态类型,在编译期确认调用哪个重载的方法,看如下代码:
interface IB {
}
interface IA extends IB {
}
class ObjG {
}
class ObjP extends ObjG {
}
public class Obj extends ObjP implements IA {
}
public static void main(String[] args){
IA obj1 = new Obj();
ObjP obj2 = new Obj();
Obj obj3 = new Obj();
}
= 左边是静态类型,右边是动态类型,即 IA, ObjP, Obj为三个对象的静态类型,确认重载方法选择的只和 = 左侧的静态类型有关。
建议对本文所有结论持怀疑态度,并自己进行验证。纸上得来终觉浅,绝知此事要躬行!