【细说Java】方法重载的简单介绍
1. 什么是重载
方法名称相同,但它们的参数类型或个数不同,这样,方法在被调用时编译器就可以根据参数的类型与个数的不同加以区分,这就是方法的重载。
既然可以通过参数类型或参数个数来作为重载条件,那返回值是否可以作为重载的条件呢?
代码如下:
public int returnType() {
return 2;
}
public float returnType() {
return 2f;
}
这时候如果这样调用此方法:
int x = returnType()
将会调用返回值是int类型的方法,这没什么好说的,如果这样调用:
returnType();
double d = returnType();
因为上面两个方法都符合条件,那么这就让编译器无法选择了,而且,方法在调用时还可以忽略其返回值。
所以方法的返回值类型不能作为重载的标准,原因就在于其不能提供给编译器足够的信息来识别应该调用哪个方法。
2. 基本类型下重载方法的调用
在重载方法调用时,如果形参列表与实参列表对应的参数类型不相同,但形参列表的参数类型可以兼容对应的实参列表的参数类型,那么这时候方法会怎么调用呢?
我们用一些简单的代码来说明:
public class Test {
public static void main(String[] args) {
Test test = new Test();
byte b = 20;
test.select(b);
} public void select(short b) {
System.out.println("调用 short方法");
} public void select(char b) {
System.out.println("调用 char方法");
} public void select(int b) {
System.out.println("调用 int方法");
} public void select(float b) {
System.out.println("调用 float方法");
} }
因为方法定义中没有形参为byte的类型,所以除了char参数的方法外,其他几个方法的参数都可以兼容byte类型,那byte会调用哪一个方法呢?
调用 short方法
同理:
public class Test {
public static void main(String[] args) {
Test test = new Test();
long b = 20L;
test.select(b);
} public void select(int b) {
System.out.println("调用 int方法");
} public void select(float b) {
System.out.println("调用 float方法");
} public void select(double b) {
System.out.println("调用 double方法");
} }
结果为:
调用 float方法
大家可以使用这几种基本数值类型多试试看。
总结:在方法调用时,如果实参的类型与方法中形参的类型不相同,那么系统会调用形参类型可以兼容的实参类型,并且形参类型与实参类型最接近的方法。而基本数据类型顺序由低到高为:
所以,加入方法调用时实参的类型为char,而要调用的方法中不存在形参为char类型的,而分别为byte,short,int,long,float,double时,byte与short不能兼容char类型,不考虑。剩下的4个方法中int与char的类型最为接近,此时就会调用形参类型为int的方法。
3. 引用类型下重载方法的调用
我们用一个简单的例子来实验:
public class Test {
public static void main(String[] args) {
Test test = new Test();
Test3 test3 = new Test3();
test.select(test3);
} public void select(Test1 test1) {
System.out.println("调用 Test1方法");
} public void select(Test2 test2) {
System.out.println("调用 Test2方法");
} public void select(Object obj) {
System.out.println("调用 Object方法");
} } class Test1 {
} class Test2 extends Test1 {
} class Test3 extends Test2 {
}
结果为:
调用 Test2方法
如果我们将参数为Test2的给去掉,则结果为:
调用 Test1方法
如果我们将参数为Test1的也给去掉,结果为:
调用 Object方法
由此可知,引用类型与基本类型的选择方式也是类似的,都是选择与实参类型最接近的。
总结:其实,不管是基本类型还是引用类型,重载方法调用时,在保证兼容性的情况下,都是会调用与实参类型最接近的方法。
4. 包装类与可变参数类型的重载方法调用
public class Test {
public static void main(String[] args) {
Test test = new Test();
test.select(5);
} public void select(float f) {
System.out.println("调用 float方法");
} public void select(double d) {
System.out.println("调用 double方法");
} public void select(Integer i) {
System.out.println("调用 Integer方法");
} public void select(Number number) {
System.out.println("调用 Number方法");
} public void select(int... i) {
System.out.println("调用 int... 方法");
} }
这种情况下,打印结果为:
调用 float方法
将float和double参数的两个方法注释掉,打印结果为:
调用 Integer方法
将前四个方法都注释掉,打印结果为:
调用 int... 方法
或许从结果中,大家也明白了包装类与可变参数类型的重载方法调用了吧;
总结:就 test.select(5)方法调用 来说:
包装类与可变参数类型的重载调用顺序,可以简记如下:
参数完全相同的方法(int) -> 形参兼容实参的方法(float -> double) -> 自动拆箱与封箱操作的方法(Integer) -> 自动拆箱与封箱的父类(Number -> Object) -> 可变参数类型的方法(int...)
5. 重载无法实现多态
看下面一段代码,并预测一下打印结果:
public class Test {
public static void main(String[] args) {
Test test = new Test();
Object obj = new Animal();
Animal animal = new Dog();
test.select(obj);
test.select(animal);
} public void select(Object number) {
System.out.println("调用 Object方法");
} public void select(Animal animal) {
System.out.println("调用 Animal 方法");
} public void select(Dog animal) {
System.out.println("调用 Dog 方法");
} } class Animal {
} class Dog extends Animal {
}
打印结果为:
调用 Object方法
调用 Animal 方法
因为重载方法的调用是在编译时确定的,也可以说是“静态绑定”,如果实参的类型是引用类型,编译器会根据引用的类型来决定调用哪个重载方法。
这种方法调用不同于多态,多态方法的调用是根据运行时对象的实际类型来决定的,也称为“动态绑定”,重载方法的选择只是根据引用的类型来决定的,而与该引用所指向的对象类型无关,因为引用所指向的对象类型是在运行时才确定的,因此,使用重载无法实现多态。
6. 泛型方法的重载
对于泛型类的参数,因为编译器在编译期间会进行类型擦除,所以和普通的重载方法稍稍有些区别。
看以下例子:
public class JavaTest {
public static void main(String[] args) {
JavaTest test = new JavaTest();
Test2 test2 = new Test2();
test.print(test2);
} public <T extends Test2> void print(T t) {
System.out.println("<T extends Test2>方法");
} public <T> void print(T t) {
System.out.println("<T>方法");
} public void print(Test1 test1) {
System.out.println("Test1方法");
}
} class Test1 {
} class Test2 extends Test1 {
}
打印结果为:
<T extends Test2>方法
有关类型擦除就说两点,其他的直接网上搜索就行:
- 将所有的泛型参数用其最左边界(最顶级的父类型)类型替换。
- 移除所有的类型参数。
如对于
public <T extends Test2> void print(T t) {
System.out.println("<T extends Test2>方法");
} public <T> void print(T t) {
System.out.println("<T>方法");
}
擦除之后的方法声明为:
public void print(Test2 t) {}
public void print(Object t) {}
然后调用规则和普通的重载方法是相同的。
参考自:《细说Java》
【细说Java】方法重载的简单介绍的更多相关文章
- java方法——重载2
什么是Java方法重载 方法重载的定义 1 对于同一个类,如果这个类里面有两个或者多个重名的方法,但是方法的参数个数.类型.顺序至少有一个不一样,这时候局构成方法重载. END 方法重载示例 1 pu ...
- Java方法重载
Java允许一个类中定义多个方法,只要参数列表不同就行了.如果同一个类中包含了两个或者两个以上的方法的方法名相同,但形参列表不同,则被称为方法重载. /* 参数类型不同的重载 */ public cl ...
- java反射机制的简单介绍
参考博客: https://blog.csdn.net/mlc1218559742/article/details/52754310 先给出反射机制中常用的几个方法: Class.forName (& ...
- Java 方法重载与引用数组类型
1.方法重载 1)方法的签名 方法的签名包含方法名和参数列表 一个类中,不可以有两个方法的签名完全相同,即一个类中不能有两个方法的方法名和参数列表都一样. public class Test{ pu ...
- Java泛型使用的简单介绍
目录 一. 泛型是什么 二. 使用泛型有什么好处 三. 泛型类 四. 泛型接口 五. 泛型方法 六. 限定类型变量 七. 泛型通配符 7.1 上界通配符 7.2 下界通配符 7.3 无限定通配符 八. ...
- Java Linked集合的简单介绍和常用方法的使用
LinkedList的简单介绍 java.util.LinkedList 集合数据存储的结构是链表结构.LinkedList是一个双向链表在实际开发中,对一个集合元素的添加和删除,经常涉及到首尾操作, ...
- Java 方法重载,方法重写(覆盖),继承等细节注意
1.方法重载(method overload)的具体规范 如果有两个方法的方法名相同,但参数不一致,那么可以说一个方法是另一个方法的重载. 一.方法名一定要相同. 二.方法的参数表必须不同,包括参数的 ...
- java方法重载和重写
1.java的方法重载和重写,表示两种不同的类型.this关键字,出现在类的构造方法中,代表使用该构造方法所创建的对象.,this可以出现在实例方法中核构造方法中.但是不能出现在类方法中.实例方法只能 ...
- Java 方法重载 方法重写
方法重载规则 参数个数不同 参数个数相同,但参数列表中对应的某个参数的类型不一样 方法的返回类型和参数名称不参与重载 "编译期绑定",,因为未产生对象,只看参数.引用类型绑定方法 ...
随机推荐
- Swift的闭包(一):闭包简介、闭包表达式的优化
定义:Closures are self-contained blocks of functionality that can be passed around and used in your co ...
- TCP/IP协议原理与应用笔记11:TCP/IP中地址与层次关系
1. 网络中常用的地址: 2. TCP/IP中地址与层次关系 :
- iOS图片拉伸
常用的图片拉伸场景有:聊天页面的气泡,需要根据内容拉伸,但圆角拉伸后会变形,为避免圆角拉伸,可以指定拉伸区域.UIImage实体调用以下方法即可指定拉伸区域. - (UIImage *)stretch ...
- 二分图最大匹配(匈牙利算法Dfs模板)
#include<iostream> #include<cstdio> #include<cstring> #define maxn 2020 using name ...
- VBA取得EXCEL表格中的行数和列数
VBA取得EXCEL表格中的行数和列数 初学EXCEL宏的童鞋,总是很想知道表格中含有数据的行数和列数,尤其是行数和列数不确定的情况下.这样可以避免很多的错误,并且可以提高效率.但每次用到的时候到网上 ...
- GIT学习(二)-->Git分布式的好处
分布式VS集中式(版本管理系统) 集中式版本控制系统,版本库是集中存放在中央服务器的,而干活的时候,用的都是自己的电脑,所以要先从中央服务器down下最新的版本,然后开始干活,干完活了,再把自己的活推 ...
- html.day01
1.web标准: 1. 结构 (xhtml) 2. 表现(css) 3.行为(js) html 超文本标记语言 xhtml (严格型超文本标记语言) 2.规范: 1. 所有标签(标记)都要 ...
- 【转】Java学习之Iterator(迭代器)的一般用法 (转)
[转]Java学习之Iterator(迭代器)的一般用法 (转) 迭代器(Iterator) 迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构.迭 ...
- Hibernate HQL查询:
Hibernate HQL查询:Criteria查询对查询条件进行了面向对象封装,符合编程人员的思维方式,不过HQL(Hibernate Query Lanaguage)查询提供了更加丰富的和灵活的查 ...
- JavaScript--垃圾回收器
垃圾回收: 释放不再被任何变量引用的对象 垃圾回收器: 专门记录对象的引用次数,并回收不再被引用的对象的程序. 垃圾回收器和主程序并行在后台执行 垃圾回收器会为每个对象创建一个引用计数器(counte ...