关于Java中的重载
发布时间:2025-05-24 19:37:21 发布人:远客网络
一、关于Java中的重载
1、含义:只要方法名相同,参数表不同,就可以构成方法重载。
2、方法重载,在JAVA中,也叫:编译时多态。(由编译的时候来确定调用哪个方法)
3、注意:如果只有print( double d)方法,而要调用 print(int),则编译器会自动地就近向上匹配 print(double)方法。但是反过来就不行,
4、因为从大到小会丢失精度,所以,没有就近向下匹配原则.
5、表示当前对象: this.age= age; this.name= name;
6、在调用本类的其它构造方法时。 This(参数表),注:this的这种用法只能放在第一行。
7、记住:在JAVA中,简单类型变量一定是传值。对象变量一这是传引用(也就是指向对象的首地址)
二、Java重载和重写
(1)方法重载是让类以统一的方式处理不同类型数据的一种手段。多个同名函数同时存在,具有不同的参数个数/类型。
重载Overloading是一个类中多态性的一种表现。
(2)java的方法重载,就是在类中可以创建多个方法,它们具有相同的名字,但具有不同的参数和不同的定义。
调用方法时通过传递给它们的不同参数个数和参数类型来决定具体使用哪个方法,这就是多态性。
(3)重载的时候,方法名要一样,但是参数类型和个数不一样,返回值类型可以相同也可以不相同。无法以返回型别作为重载函数的区分标准。
//这是这个程序的第一种编程方法,在main方法中先创建一个Dog类实例,然后在Dog类的构造方法中利用this关键字调用不同的bark方法。
不同的重载方法bark是根据其参数类型的不同而区分的。
//注意:除构造器以外,编译器禁止在其他任何地方中调用构造器。
voidbark()//bark()方法是重载方法
System.out.println(\"nobarking!\");
voidbark(Stringm,doublel)//注意:重载的方法的返回值都是一样的,
System.out.println(\"abarkingdog!\");
voidbark(inta,Stringn)//不能以返回值区分重载方法,而只能以“参数类型”和“类名”来区分
System.out.println(\"ahowlingdog\");
publicstaticvoidmain(String[]args)
//dog.bark(\"male\",\"yellow\");
然后我们再来谈谈重写(Overriding)
(1)父类与子类之间的多态性,对父类的函数进行重新定义。如果在子类中定义某方法与其父类有相同的名称和参数,我们说该方法被重写(Overriding)。在Java中,子类可继承父类中的方法,而不需要重新编写相同的方法。
但有时子类并不想原封不动地继承父类的方法,而是想作一定的修改,这就需要采用方法的重写。
(2)若子类中的方法与父类中的某一方法具有相同的方法名、返回类型和参数表,则新方法将覆盖原有的方法。
如需父类中原有的方法,可使用super关键字,该关键字引用了当前类的父类。
(3)子类函数的访问修饰权限不能少于父类的;
1、编译器检查对象声明的类型和方法名,从而获取所有候选方法。试着把上例Base类的test注释掉,这时再编译就无法通过。
2、重载决策:编译器检查方法调用的参数类型,从上述候选方法选出唯一的那一个(其间会有隐含类型转化)。
如果编译器找到多于一个或者没找到,此时编译器就会报错。试着把上例Base类的test(byteb)注释掉,这时运行结果是11。
3、若方法类型为priavtestaticfinal,java采用静态编译,编译器会准确知道该调用哪个方法。
4、当程序运行并且使用动态绑定来调用一个方法时,那么虚拟机必须调用对象的实际类型相匹配的方法版本。
在例子中,云南java课程发现b所指向的实际类型是TestOverriding,所以b.test(0)调用子类的test。
但是,子类并没有重写test(byteb),所以b.test((byte)0)调用的是父类的test(byteb)。
如果把父类的(byteb)注释掉,则通过第二步隐含类型转化为int,最终调用的是子类的test(inti)。
三、Java中覆盖和重载的区别
override可以翻译为覆盖,从字面就可以知道,它是覆盖了一个方法并且对其重写,以求达到不同的作用。最熟悉的覆盖就是对接口方法的实现,在接口中一般只是对方法进行了声明,而在实现时,就需要实现接口声明的所有方法。除了这个典型的用法以外,在继承中也可能会在子类覆盖父类中的方法。在覆盖要注意以下的几点:
1、覆盖的方法的标志必须要和被覆盖的方法的标志完全匹配,才能达到覆盖的效果;
2、覆盖的方法的返回值必须和被覆盖的方法的返回一致;
3、覆盖的方法所抛出的异常必须和被覆盖方法的所抛出的异常一致,或者是其子类;
4、被覆盖的方法不能为private,否则在其子类中只是新定义了一个方法,并没有对其进行覆盖。
overload对来说可能比较熟悉,可以翻译为重载,它是指可以定义一些名称相同的方法,通过定义不同的输入参数来区分这些方法,然后再调用时,VM就会根据不同的参数样式,来选择合适的方法执行。在使用重载要注意以下的几点:
1、在使用重载时只能通过不同的参数样式。例如,不同的参数类型,不同的参数个数,不同的参数顺序(当然,同一方法内的几个参数类型必须不一样,例如可以是fun(int, float),但是不能为fun(int, int));
2、不能通过访问权限、返回类型、抛出的异常进行重载;
3、方法的异常类型和数目不会对重载造成影响;
4、对于继承来说,如果某一方法在父类中是访问权限是priavte,那么就不能在子类对其进行重载,如果定义的话,也只是定义了一个新方法,而不会达到重载的效果。
下面是对override和overload的测试程序,其中注释中的内容都是会产生编译错误的代码,将注释去掉,看看在编译时会产生什么效果。
//对overload测试的文件:OverloadTest.java
//下面几个方法用来验证可以通过定义不同的参数类型和参数的数目进行方法重载。
System.out.println("method fun in OverloadTest, no parameter");
System.out.println("method fun in OverloadTest, parameter type: float");
System.out.println("method fun in OverloadTest, parameter type: int");
public void fun(int i1, int i2){
System.out.println("method fun in OverloadTest, parameter type: int, int");
//下面的两个方法用来验证可以通过定义不同的参数顺序进行方法重载。
//需要注意:这里的参数肯定不是相同的类型,否则的顺序的先后就毫无意义。
public void fun1(int i, float f){
System.out.println("method fun1 in OverloadTest, sequence of parameters is: int, float");
public void fun1(float f, int i){
System.out.println("method fun1 in OverloadTest, sequence of parameters is: float, int");
//下面的两个方法用来验证方法抛出的异常对于重载的影响.
//无论是异常的类型还是异常的个数都不会对重载造成任何的影响。
public void fun2() throws TestException{
System.out.println("fun2 in OverloadTest, exception: TestException");
public void fun2(int i) throws TestException, TestException1{
System.out.println("fun2 in OverloadTest, exception: TestException, TestException1");
public void fun2(float f) throws Exception{
System.out.println("fun2 in OverloadTest, exception: Exception");
//不能通过抛出的异常类型来重载fun方法。
//public void fun(int i) throws Exception{
// System.out.println("method fun in OverloadTest, parameter type: int, exception: Exception");
//public boolean fun(int i) throws Exception{
// System.out.println("method fun in OverloadTest, parameter type: int, exception: Exception, return: boolean");
//不能通过不同的访问权限进行重载
public static void main(String[] args){
//这里只是定义了OverloadTest的实例,所以test不会调用
OverloadTest test= new OverloadTest1();
//这里定义了OverloadTest1的实例,因为OverloadTest1是OverloadTest
//的子类,所以test1会调用OverloadTest中的方法。
OverloadTest1 test1= new OverloadTest1();
//这里不会调用OverloadTest1的fun方法
//这个调用不会执行,因为fun3()在OverloadTest中访问权限是priavte
class OverloadTest1 extends OverloadTest{
public void fun(int i, int m, int n){
System.out.println("Overload fun1 in OverloadTest1, parameter type: int, int, int");
//这个不是对父类中方法的重载,只是一个新的方法。
System.out.println("fun2 in OverloadTest1");
//对override测试的文件:OverrideTest.java
public void fun() throws TestException{
System.out.println("method fun in OverrideTest");
System.out.println("method fun1 in OverrideTest");
public static void main(String[] args){
OverrideTest test= new OverrideTest1();
class OverrideTest1 extends OverrideTest{
public void fun() throws TestException2{
System.out.println("fun in OverrideTest1");
//不能Override父类中的方法,因为它定义了不同的异常类型和
//public int fun() throws TestException1{
// System.out.println("method fun in Test");
//不能Override父类中的方法,因为它抛出了比父类中非法范围
//public void fun() throws Exception{
// System.out.println("fun in OverrideTest1");
//这个方法并没有Override父类中的fun1方法,因为这个方法在
//父类是private类型,所以这里只是相当于定义了一个新方法。
System.out.println("method fun1 in Test");
class TestException extends Exception{
public TestException(String msg){
class TestException1 extends TestException{
public TestException1(String msg){
class TestException2 extends TestException{
public TestException2(String msg){