在Java中为什么不同的返回类型不算方法重载?
本文已收录《Java常见面试题》:https://gitee.com/mydb/interview
方法重载是指在同一个类中,定义了多个同名方法,但每个方法的参数类型或者是参数个数不同就是方法重载。
比如以下 4 个 method 方法就可以称之为方法重载,如下代码所示:
public class OverloadExample {
public void method() {
// doSomething
}
public void method(String name) {
// doSomething
}
public void method(Integer id) {
// doSomething
}
public void method(Integer id, String name) {
// doSomething
}
}
为什么不同返回类型不算方法重载?
要回答这个问题,首先要了解一点前置内容,方法签名。
方法签名是由:方法名称 + 参数类型 + 参数个数组成的一个唯一值,这个唯一值就是方法签名,而 JVM(Java 虚拟机)就是通过这个方法签名来决定调用哪个方法的。
从方法签名的组成规则我们可以看出,方法的返回类型不是方法签名的组成部分,所以当同一个类中出现了多个方法名和参数相同,但返回值类型不同的方法时,JVM 就没办法通过方法签名来判断到底要调用哪个方法了,如下图所示:
那为什么返回类型不能做为方法签名的一部分呢?
原因其实很简单,试想一下,如果方法的返回类型也作为方法签名的一部分,那么当程序员写了一个代码去调用“重载”的方法时,JVM 就不能分辨要调用哪个方法了,如下代码所示:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method("磊哥"); // JVM 应该调用哪个方法?
}
public int method(String name) {
// doSomething
return 666;
}
public String method(String name) {
// doSomething
return "磊哥聊编程";
}
}
像以上情况,JVM 就推断不出来要调用哪个方法了,所以方法的返回类型不能作为方法签名的一部分。
方法重载的使用场景
方法重载的经典使用场景是 String 类型的 valueOf 方法,valueOf 方法重载有 9 种实现,如下图所示:
它可以将数组、对象和基础数据类型转换成字符串类型。
方法重载匹配原则
方法重载的调用顺序是有前后之分的,比如以下代码:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method(12);
}
public void method(int num) {
System.out.println("调用 int 方法");
}
public void method(long num) {
System.out.println("调用 long 方法");
}
public void method(Integer num) {
System.out.println("调用 Integer 方法");
}
public void method(Object num) {
System.out.println("调用 Object 方法");
}
public void method(int... num) { // 可选参数
System.out.println("调用 int... 方法");
}
}
当出现方法重载时,程序要调用哪个方法呢?执行以上程序的执行结果如下:
因此我们可以得出以下结论。
匹配原则1:精准类型匹配
方法重载会优先调用和方法参数类型一模一样的方法,这是第一优先匹配原则:精准类型匹配。
匹配原则2:基本类型自动转换成更大的基本类型
接下来我们把精准匹配方法删掉,观察一下第二匹配顺序是什么?实现代码如下:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method(12);
}
public void method(long num) {
System.out.println("调用 long 方法");
}
public void method(Integer num) {
System.out.println("调用 Integer 方法");
}
public void method(Object num) {
System.out.println("调用 Object 方法");
}
public void method(int... num) { // 可选参数
System.out.println("调用 int... 方法");
}
}
以上程序的执行结果如下图所示:
因此我们可以得出结论:如果是基本数据类型,那么方法重载调用的第二匹配原则是自动转换成更大的基本数据类型。
匹配原则3:自动装/拆箱匹配
接下来将第二匹配原则中的 long 方法也删除掉,实现代码如下:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method(12);
}
public void method(Integer num) {
System.out.println("调用 Integer 方法");
}
public void method(Object num) {
System.out.println("调用 Object 方法");
}
public void method(int... num) { // 可选参数
System.out.println("调用 int... 方法");
}
}
以上程序的执行结果如下图所示:
从上述执行结果可以看出,方法重载的第三匹配原则是,匹配自动装箱或拆箱的数据类型。
匹配原则4:按照继承路线依次向上匹配
此时将第三匹配原则中的 Integer 方法删除,剩下代码如下:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method(12);
}
public void method(Object num) {
System.out.println("调用 Object 方法");
}
public void method(int... num) { // 可选参数
System.out.println("调用 int... 方法");
}
}
以上程序的执行结果如下图所示:
从上述执行结果可以看出,方法重载的第四匹配原则是,依次向上匹配父类的方法调用。
匹配原则5:可变参数匹配
最后将代码中的方法删除的只剩一个可选参数,实现代码如下:
public class OverloadExample {
public static void main(String[] args) {
OverloadExample example = new OverloadExample();
example.method(12);
}
public void method(int... num) { // 可选参数
System.out.println("调用 int... 方法");
}
}
以上程序的执行结果如下图所示:
从上述执行结果可以看出,方法重载的第五匹配原则是,匹配可选参数。
总结
在同一个类中定义了多个同名方法,但每个方法的参数类型或者是参数个数不同就是方法重载。方法重载的典型使用场景是 String 中的 valueOf 方法,它有 9 种实现。方法返回类型不能作为方法重载的依据,因为它不是方法签名的组成部分。方法重载有 5 个匹配原则:精准匹配、基本类型自动转换成更大的基本类型匹配、自动装/拆箱匹配、按照继承路线依次向上匹配、可变参数匹配。
参考资料:《码出高效》
是非审之于己,毁誉听之于人,得失安之于数。
公众号:Java面试真题解析
在Java中为什么不同的返回类型不算方法重载?的更多相关文章
- java中相同名字不同返回类型的方法
这种名字相同返回类型不同的方法,在同一个类中是无法共存的,不论是继承过来的方法,还是多实现过来的方法,在一个类内都无法共存.名字确定了,你能改的只有参数(重载).
- (转) Java中的负数及基本类型的转型详解
(转) https://my.oschina.net/joymufeng/blog/139952 面这行代码的输出是什么? 下面两行代码的输出相同吗? 请尝试在Eclipse中运行上面的两个代码片段, ...
- 【转】java中byte数组与int类型的转换(两种方式)----不错
原文网址:http://blog.csdn.net/piaojun_pj/article/details/5903009 java中byte数组与int类型的转换,在网络编程中这个算法是最基本的算法, ...
- java中读取特殊文件的类型
java中读取特殊文件的类型: 第一种方法(字符拼接读取): public static String getType(String s){ String s1=s.substring(s.index ...
- Java中的基本类型和包装类型区别
首先看一下几个测试题,验证一下java中对基本类型和包装类型的理解,看看最后输出的答案对不对,答案在这篇博客中哦: // 第一题: 基本类型和包装类型 int a = 100; Integer b = ...
- Java中的两种异常类型及其区别?
Java中的两种异常类型是什么?他们有什么区别? Throwable包含了错误(Error)和异常(Excetion两类) Exception又包含了运行时异常(RuntimeException, 又 ...
- JAVA中如何获取变量的类型
JAVA中如何获取变量的类型? package xiya; public class Demo { public static void main(String[] args) { String ty ...
- java中如何理解:其他类型 + string 与 自增类型转换和赋值类型转换
java中如何理解:其他类型 + string 与 自增类型转换和赋值类型转换 一.字符串与其他类型连接 public class DemoString{ public static void mai ...
- Java执行shell脚本并返回结果两种方法的完整代码
Java执行shell脚本并返回结果两种方法的完整代码 简单的是直接传入String字符串,这种不能执行echo 或者需要调用其他进程的命令(比如调用postfix发送邮件命令就不起作用) 执行复杂的 ...
随机推荐
- MyBatis(4):使用limit实现分页
用limit实现分页,首先要创建一个Maven项目,搭建好mybatis的实验环境,并且连接好数据库 代码 1,编写dao接口 UserMapper //查询全部用户实现分页 List<User ...
- 【Linux】【Services】【MessageQueue】搭建高可用rabbitMQ
1. 简介 1.1. 官方网站: https://www.rabbitmq.com/ 1.2. 配置文档:https://docs.openstack.org/ha-guide/shared-mess ...
- 【Service】【Web】【Middleware】Tomcat
1. 概念 1.1. 官方网站:tomcat.apache.org 1.2. tomcat的组件 <Server> <Service> <Connector/> & ...
- springmvc框架找那个@responseBody注解
<%@ page contentType="text/html;charset=UTF-8" language="java" %><html& ...
- JVM堆空间结构及常用的jvm内存分析命令和工具
jdk8之前的运行时数据区域 程序计数器 是一块较小的内存空间,它可以看做是当前线程所执行的字节码的行号指示器.每个线程都有一个独立的程序计数器,这类内存区域为"线程私有",此内存 ...
- Linux系统的文件复制移动删除与VIM编辑
目录 今日内容概要 内容详细 复制文件 移动文件 删除文件 系统别名(针对 rm 改别名) vim编辑器 今日内容概要 复制文件 移动文件 删除文件 vim编辑器 内容详细 复制文件 # 命令: cp ...
- ctypes与numpy.ctypeslib的使用
numpy ctypeslib 与 ctypes接口使用说明 作者:elfin 目录 一.numpy.ctypeslib使用说明 1.1 准备好一个C++计算文件 1.2 ctypeslib主要的五个 ...
- 从一次解决Nancy参数绑定“bug”开始发布自己的第一个nuget包(上篇)
起因 最近,同事跟我说,他们负责的一个Api程序出现了一些很奇怪的事情.这个Api是为环保局做的一个扬尘质控大屏提供数据的,底层是基于Nancy做的.因为发现有些接口的数据出现异常,他就去调试了一下, ...
- Jsp/Servlet文件的上传和下载
文件上传的入门 文件上传的步骤: 总结实现思路: 1.创建核心上传类ServletFileUpload,这个时候需要一个工厂类 2.创建磁盘工厂类对象DiskFileItemFactory ...
- 如何查看Python的版本号
一.如何查看Python的版本号 win+r输入cmd在输入:python --version回车即可