2-4Java重写与重载

Java 重写与重载

重写(Override)

重写是子类对父类的允许访问的方法的实现过程进行重新编写, 返回值和形参都不能改变。即外壳不变,核心重写!

重写的好处在于子类可以根据需要,定义特定于自己的行为。 也就是说子类能够根据需要实现父类的方法。

重写方法不能抛出新的检查异常或者比被重写方法申明更加宽泛的异常。例如: 父类的一个方法申明了一个检查异常 IOException,但是在重写这个方法的时候不能抛出 Exception 异常,因为 Exception 是 IOException 的父类,只能抛出 IOException 的子类异常。

在面向对象原则里,重写意味着可以重写任何现有方法。实例如下:

`class` `Animal{``  ``public` `void` `move(){``   ``System.out.println(``"动物可以移动"``);``  ``}``}` `class` `Dog ``extends` `Animal{``  ``public` `void` `move(){``   ``System.out.println(``"狗可以跑和走"``);``  ``}``}` `public` `class` `TestDog{``  ``public` `static` `void` `main(String args[]){``   ``Animal a = ``new` `Animal(); ``// Animal 对象``   ``Animal b = ``new` `Dog(); ``// Dog 对象` `   ``a.move();``// 执行 Animal 类的方法` `   ``b.move();``//执行 Dog 类的方法``  ``}``}`

以上实例编译运行结果如下:

`动物可以移动``狗可以跑和走`

在上面的例子中可以看到,尽管b属于Animal类型,但是它运行的是Dog类的move方法。

这是由于在编译阶段,只是检查参数的引用类型。

然而在运行时,Java虚拟机(JVM)指定对象的类型并且运行该对象的方法。

因此在上面的例子中,之所以能编译成功,是因为Animal类中存在move方法,然而运行时,运行的是特定对象的方法。

思考以下例子:

`class` `Animal{``  ``public` `void` `move(){``   ``System.out.println(``"动物可以移动"``);``  ``}``}` `class` `Dog ``extends` `Animal{``  ``public` `void` `move(){``   ``System.out.println(``"狗可以跑和走"``);``  ``}``  ``public` `void` `bark(){``   ``System.out.println(``"狗可以吠叫"``);``  ``}``}` `public` `class` `TestDog{``  ``public` `static` `void` `main(String args[]){``   ``Animal a = ``new` `Animal(); ``// Animal 对象``   ``Animal b = ``new` `Dog(); ``// Dog 对象` `   ``a.move();``// 执行 Animal 类的方法``   ``b.move();``//执行 Dog 类的方法``   ``b.bark();``  ``}``}`

以上实例编译运行结果如下:

`TestDog.java:``30``: cannot find symbol``symbol : method bark()``location: ``class` `Animal``        ``b.bark();``         ``^`

该程序将抛出一个编译错误,因为b的引用类型Animal没有bark方法。

方法的重写规则

  • 参数列表必须完全与被重写方法的相同。
  • 返回类型与被重写方法的返回类型可以不相同,但是必须是父类返回值的派生类(java5 及更早版本返回类型要一样,java7 及更高版本可以不同)。
  • 访问权限不能比父类中被重写的方法的访问权限更低。例如:如果父类的一个方法被声明为 public,那么在子类中重写该方法就不能声明为 protected。
  • 父类的成员方法只能被它的子类重写。
  • 声明为 final 的方法不能被重写。
  • 声明为 static 的方法不能被重写,但是能够被再次声明。
  • 子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为 private 和 final 的方法。
  • 子类和父类不在同一个包中,那么子类只能够重写父类的声明为 public 和 protected 的非 final 方法。
  • 重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。
  • 构造方法不能被重写。
  • 如果不能继承一个方法,则不能重写这个方法。

Super 关键字的使用

当需要在子类中调用父类的被重写方法时,要使用 super 关键字。

`class` `Animal{``  ``public` `void` `move(){``   ``System.out.println(``"动物可以移动"``);``  ``}``}` `class` `Dog ``extends` `Animal{``  ``public` `void` `move(){``   ``super``.move(); ``// 应用super类的方法``   ``System.out.println(``"狗可以跑和走"``);``  ``}``}` `public` `class` `TestDog{``  ``public` `static` `void` `main(String args[]){` `   ``Animal b = ``new` `Dog(); ``// Dog 对象``   ``b.move(); ``//执行 Dog类的方法` `  ``}``}`

以上实例编译运行结果如下:

`动物可以移动``狗可以跑和走`

重载(Overload)

重载(overloading) 是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同。

每个重载的方法(或者构造函数)都必须有一个独一无二的参数类型列表。

最常用的地方就是构造器的重载。

重载规则:

  • 被重载的方法必须改变参数列表(参数个数或类型不一样);
  • 被重载的方法可以改变返回类型;
  • 被重载的方法可以改变访问修饰符;
  • 被重载的方法可以声明新的或更广的检查异常;
  • 方法能够在同一个类中或者在一个子类中被重载。
  • 无法以返回值类型作为重载函数的区分标准。

实例

`public` `class` `Overloading {``  ``public` `int` `test(){``    ``System.out.println(``"test1"``);``    ``return` `1``;``  ``}` `  ``public` `void` `test(``int` `a){``    ``System.out.println(``"test2"``);``  ``} ` `  ``//以下两个参数类型顺序不同``  ``public` `String test(``int` `a,String s){``    ``System.out.println(``"test3"``);``    ``return` `"returntest3"``;``  ``} ` `  ``public` `String test(String s,``int` `a){``    ``System.out.println(``"test4"``);``    ``return` `"returntest4"``;``  ``} ` `  ``public` `static` `void` `main(String[] args){``    ``Overloading o = ``new` `Overloading();``    ``System.out.println(o.test());``    ``o.test(``1``);``    ``System.out.println(o.test(``1``,``"test3"``));``    ``System.out.println(o.test(``"test4"``,``1``));``  ``}``}`

重写与重载之间的区别

区别点 重载方法 重写方法
参数列表 必须修改 一定不能修改
返回类型 可以修改 一定不能修改
异常 可以修改 可以减少或删除,一定不能抛出新的或者更广的异常
访问 可以修改 一定不能做更严格的限制(可以降低限制)

方法的重写(Overriding)和重载(Overloading)是java多态性的不同表现,重写是父类与子类之间多态性的一种表现,重载可以理解成多态的具体表现形式。

  1. 方法重载是一个类中定义了多个方法名相同,而他们的参数的数量不同或数量相同而类型和次序不同,则称为方法的重载(Overloading)。
  2. 方法重写是在子类存在方法与父类的方法的名字相同,而且参数的个数与类型一样,返回值也一样的方法,就称为重写(Overriding)。
  3. 方法重载是一个类的多态性表现,而方法重写是子类与父类的一种多态性表现。

1725873027201

给大家分享有些细节(自己总结的):

1.实例方法就是成员方法,就是在类里面没有被static修饰的方法,而被static修饰的方法叫做静态方法,静态方法属于类,可以直接用类名调用,在静态方法里面不可以直接用实例方法即成员方法,主入口函数(public static void main()就是一个静态方法,所以如果想要调用类里面的成员方法需要先实例对应的类,才可以调用相应的方法。2.final修饰的方法不能重写,类方法和构造方法都不可以重写,被final修饰的变量,因为不可以被修改故一般用于创建常量,static final修饰的方法的变量叫做类常量,可以用类直接引用,被final修饰的类叫做最终类,不可以被继承,因为被fianl修饰的类一般用于不被别的类拓展,不想被别人看见,但是可以继承别的没有被final修饰的类。3.重写的方法的访问修饰符(public,protected,private,default)要比被重写的方法的范围要大,就和继承一样,子类的访问修饰符的范围要比父类的大或者相等,且返回类型,参数列表和名字要一样,如果列表参数不一样就是重载了。4.重载就是访问修饰符没有限制(随意都可以),返回类型可以不一样,名字要一样,但是参数列表一定要不一样(参数的数量不一样,类型和次序不一样都可以呢),而且重载的方法可以写在子类里面重载呢 5.接口里面的变量默认都是public static final 修饰的,接口里面的方法都是public abstract,也就是接口里面的方法默认是抽象类,但是没有实现方法哈,如果想要有实现方法可以用default修饰,代表默认方法,提示:接口里面的方法只能用public修饰,因为接口里面的方法是公开的,要不然会报错哈。6.在上面用到了多态,即上面的Animal b=new Dog();这里就是说呢,创建了一个引用类型是Animal,实例对象是Dog类的一个对象b,多态就是用来调用子类中重写方法的一个方法,但是不可以调用子类的特有方法,比如上题的bark方法(子类)。7.super关键字可以用来初始化父类的构造方法和调用父类的被重写方法super.mannername,但是不可以在静态方法里面super。

posted @   奋斗的独角兽  阅读(9)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 从HTTP原因短语缺失研究HTTP/2和HTTP/3的设计差异
· 三行代码完成国际化适配,妙~啊~
点击右上角即可分享
微信分享提示