Java 教程
- Java - 首页
- Java - 概述
- Java - 历史
- Java - 特性
- Java vs. C++
- JVM - Java虚拟机
- Java - JDK vs JRE vs JVM
- Java - Hello World 程序
- Java - 环境搭建
- Java - 基本语法
- Java - 变量类型
- Java - 数据类型
- Java - 类型转换
- Java - Unicode 系统
- Java - 基本运算符
- Java - 注释
- Java - 用户输入
- Java - 日期与时间
Java 控制语句
- Java - 循环控制
- Java - 决策制定
- Java - if-else
- Java - switch
- Java - for 循环
- Java - for-each 循环
- Java - while 循环
- Java - do-while 循环
- Java - break
- Java - continue
面向对象编程
- Java - OOPs 概念
- Java - 对象与类
- Java - 类属性
- Java - 类方法
- Java - 方法
- Java - 变量作用域
- Java - 构造函数
- Java - 访问修饰符
- Java - 继承
- Java - 聚合
- Java - 多态
- Java - 方法重写
- Java - 方法重载
- Java - 动态绑定
- Java - 静态绑定
- Java - 实例初始化块
- Java - 抽象
- Java - 封装
- Java - 接口
- Java - 包
- Java - 内部类
- Java - 静态类
- Java - 匿名类
- Java - 单例类
- Java - 包装类
- Java - 枚举
- Java - 枚举构造函数
- Java - 枚举字符串
Java 内置类
Java 文件处理
Java 错误与异常
- Java - 异常
- Java - try-catch 块
- Java - try-with-resources
- Java - 多重catch块
- Java - 嵌套try块
- Java - finally 块
- Java - throw 异常
- Java - 异常传播
- Java - 内置异常
- Java - 自定义异常
Java 多线程
- Java - 多线程
- Java - 线程生命周期
- Java - 创建线程
- Java - 启动线程
- Java - 线程连接
- Java - 线程命名
- Java - 线程调度器
- Java - 线程池
- Java - 主线程
- Java - 线程优先级
- Java - 守护线程
- Java - 线程组
- Java - 关闭钩子
Java 同步
Java 网络编程
- Java - 网络编程
- Java - Socket 编程
- Java - URL 处理
- Java - URL 类
- Java - URLConnection 类
- Java - HttpURLConnection 类
- Java - Socket 类
- Java -泛型
Java 集合
Java 接口
Java 数据结构
Java 集合算法
高级 Java
- Java - 命令行参数
- Java - Lambda 表达式
- Java - 发送邮件
- Java - Applet 基础
- Java - Javadoc 注释
- Java - 自动装箱和拆箱
- Java - 文件不匹配方法
- Java - REPL (JShell)
- Java - 多版本 Jar 文件
- Java - 私有接口方法
- Java - 内部类菱形运算符
- Java - 多分辨率图像 API
- Java - 集合工厂方法
- Java - 模块系统
- Java - Nashorn JavaScript
- Java - Optional 类
- Java - 方法引用
- Java - 函数式接口
- Java - 默认方法
- Java - Base64 编码解码
- Java - switch 表达式
- Java - Teeing 收集器
- Java - 微基准测试
- Java - 文本块
- Java - 动态 CDS 归档
- Java - Z 垃圾收集器 (ZGC)
- Java - 空指针异常
- Java - 打包工具
- Java - 密封类
- Java - 记录类
- Java - 隐藏类
- Java - 模式匹配
- Java - 简洁的数字格式化
- Java - 垃圾回收
- Java - JIT 编译器
Java 杂项
- Java - 递归
- Java - 正则表达式
- Java - 序列化
- Java - 字符串
- Java - Process API改进
- Java - Stream API改进
- Java - 增强型@Deprecated 注释
- Java - CompletableFuture API改进
- Java - 流
- Java - 日期时间 API
- Java 8 - 新特性
- Java 9 - 新特性
- Java 10 - 新特性
- Java 11 - 新特性
- Java 12 - 新特性
- Java 13 - 新特性
- Java 14 - 新特性
- Java 15 - 新特性
- Java 16 - 新特性
Java API 和框架
Java 类引用
- Java - Scanner
- Java - 数组
- Java - 字符串
- Java - Date
- Java - ArrayList
- Java - Vector
- Java - Stack
- Java - PriorityQueue
- Java - LinkedList
- Java - ArrayDeque
- Java - HashMap
- Java - LinkedHashMap
- Java - WeakHashMap
- Java - EnumMap
- Java - TreeMap
- Java - IdentityHashMap
- Java - HashSet
- Java - EnumSet
- Java - LinkedHashSet
- Java - TreeSet
- Java - BitSet
- Java - Dictionary
- Java - Hashtable
- Java - Properties
- Java - Collection
- Java - Array
Java 有用资源
Java - 方法重写
在上一章中,我们讨论了超类和子类。如果一个类继承了其超类的方法,那么就有可能重写该方法,前提是该方法没有被标记为 final。
Java 中重写的益处
重写的益处是:能够定义特定于子类类型的行为,这意味着子类可以根据其需求实现父类方法。
Java 方法重写
方法重写允许我们实现运行时多态性,并用于编写已经在超类中定义的子类方法的特定定义。
超类中的方法和子类中被重写的方法应该具有相同的声明签名,例如参数列表、类型和返回类型。
Java 方法重写的用法
以下是 Java 中方法重写的两个重要用法:
- 方法重写用于实现运行时多态性。
- 方法重写用于编写子类方法的特定定义(此方法称为被重写方法)。
Java 方法重写的示例
让我们来看一个例子。
class Animal { public void move() { System.out.println("Animals can move"); } } class Dog extends Animal { public void move() { System.out.println("Dogs can walk and run"); } } public class TestDog { public static void main(String args[]) { Animal a = new Animal(); // Animal reference and object Animal b = new Dog(); // Animal reference but Dog object a.move(); // runs the method in Animal class b.move(); // runs the method in Dog class } }
输出
Animals can move Dogs can walk and run
在上面的例子中,你可以看到,即使b是 Animal 类型,它也会运行 Dog 类中的 move 方法。原因是:在编译时,检查是在引用类型上进行的。但是,在运行时,JVM 会确定对象类型,并运行属于该特定对象的方法。
因此,在上面的示例中,程序将正确编译,因为 Animal 类具有 move 方法。然后,在运行时,它会运行特定于该对象的方法。
考虑以下示例:
示例
class Animal { public void move() { System.out.println("Animals can move"); } } class Dog extends Animal { public void move() { System.out.println("Dogs can walk and run"); } public void bark() { System.out.println("Dogs can bark"); } } public class TestDog { public static void main(String args[]) { Animal a = new Animal(); // Animal reference and object Animal b = new Dog(); // Animal reference but Dog object a.move(); // runs the method in Animal class b.move(); // runs the method in Dog class b.bark(); } }
输出
TestDog.java:26: error: cannot find symbol b.bark(); ^ symbol: method bark() location: variable b of type Animal 1 error
此程序将抛出编译时错误,因为 b 的引用类型 Animal 没有名为 bark 的方法。
方法重写的规则
参数列表必须与被重写方法的参数列表完全相同。
返回类型应该与超类中原始被重写方法声明的返回类型相同或为其子类型。
访问级别不能比被重写方法的访问级别更严格。例如:如果超类方法声明为 public,则子类中的重写方法不能是 private 或 protected。
实例方法只有在被子类继承时才能被重写。
声明为 final 的方法不能被重写。
声明为 static 的方法不能被重写,但可以被重新声明。
如果一个方法不能被继承,那么它就不能被重写。
与实例的超类位于同一包中的子类可以重写任何未声明为 private 或 final 的超类方法。
不同包中的子类只能重写声明为 public 或 protected 的非 final 方法。
重写方法可以抛出任何未检查异常,无论被重写方法是否抛出异常。但是,重写方法不应抛出比被重写方法声明的异常更新或更广泛的已检查异常。重写方法可以抛出比被重写方法更窄或更少的异常。
构造函数不能被重写。
Java 方法和构造函数重写
在 Java 中,每个类都有不同的名称,构造函数的名称与类名相同。因此,我们不能重写构造函数,因为它们不能具有相同的名称。
Java 方法重写:使用 super 关键字
调用被重写方法的超类版本时,使用super关键字。
示例:使用 super 关键字
class Animal { public void move() { System.out.println("Animals can move"); } } class Dog extends Animal { public void move() { super.move(); // invokes the super class method System.out.println("Dogs can walk and run"); } } public class TestDog { public static void main(String args[]) { Animal b = new Dog(); // Animal reference but Dog object b.move(); // runs the method in Dog class } }
输出
Animals can move Dogs can walk and run