一、方法的重载
方法名一样,但参数不一样,这就是重载(overload)。
所谓的参数不一样,主要有两点:第一是参数的个数不一样,第二是参数的类型不一样。只要这两方面有其中的一方面不一样就可以构成方法的重载了。
1 package cn.galc.test; 2 3 public class TestOverLoad { 4 5 void max(int a, int b) { 6 System.out.println(a > b ? a : b); 7 } 8 9 /* 10 * int max(int a, int b) { 11 * return a > b ? a : b; 12 * } 13 */ 14 15 void max(float a, float b) { 16 System.out.println(a > b ? a : b); 17 } 18 }
这里有void修饰符的两个max方法名字相同,但它们的参数类型不一样,所以可以构成重载。而int max(int a , int b)方法与void max(int a, int b)方法是不构成重载的,现在它们是重名的两个方法,在一个类中声明两个重名的方法是不允许的,编译会出错。方法名一样,参数类型一样,只有返回值不一样,这个不构成重载,因为方法可以这么调用,调用一个方法时可以不使用的它返回值,所以当调用这两个方法时,把整型的数传进去,由于两个方法都是一样的名字,一样的参数类型,所以编译器无法区分到底要调用的是哪一个方法。构成重载深层次的原因:只要这两个方法编译器能区分开来,在调用的时候知道要调用的是哪一个,不会产生混淆,这两个方法就构成重载。
再看下面这两个方法:
1 int max(int a, int b) { 2 System.out.println("调用的int max(int a, int b)方法"); 3 return a > b ? a : b; 4 } 5 6 int max(short a, short b) { 7 System.out.println("调用的int max(short a, short b)方法"); 8 return a > b ? a : b; 9 }
这两个方法也是可以构成重载的,因为编译器一看到整数,就会把它当作int类型。所以当把整数传进来的时候,编译器首先调用的是max(int a , int b)这个方法。而要想调用max(short a ,short b)这个方法,那么就得在main方法里面这样写:
1 public static void main(String[] args) { 2 TestOverLoad t= new TestOverLoad(); 3 t.max(3,4); //这里调用的是max(int a , int b)这个方法 4 short a = 3; 5 short b = 4; 6 t.max(a, b); //这里调用的是max(short a , short b)这个方法。 7 }
二、构造方法的重载
与普通方法一样,构造方法也可以重载
1 package cn.galc.test; 2 3 public class Person { 4 5 int id; 6 int age; 7 8 /** 9 * 构造方法 10 */ 11 public Person() { 12 id=0; 13 age=20; 14 } 15 16 /** 17 * 构造方法重载一 18 * @param i 19 */ 20 public Person(int i) { 21 id=i; 22 age=20; 23 } 24 25 /** 26 * 构造方法重载二 27 * @param i 28 * @param j 29 */ 30 public Person(int i,int j) { 31 id=i; 32 age=j; 33 } 34 }
三、重写&应用
- 子父类中成员方法的特点
当在程序中通过对象调用方法时,会先在子类中查找有没有对应的方法,若子类中存在就会执行子类中的方法,若子类中不存在就会执行父类中相应的方法。
看如下代码:
class Fu{
public void show(){
System.out.println("Fu类中的show方法执行");
}
}
class Zi extends Fu{
public void show2(){
System.out.println("Zi类中的show2方法执行");
}
}
public class Test{
public static void main(String[] args) {
Zi z = new Zi();
z.show(); //子类中没有show方法,但是可以找到父类方法去执行
z.show2();
}
}
- 成员方法特殊情况——覆盖
子类中出现与父类一模一样的方法时,会出现覆盖操作,也称为override重写、复写或者覆盖。
class Fu
{
public void show()
{
System.out.println("Fu show");
}
}
class Zi extends Fu
{
//子类复写了父类的show方法
public void show()
{
System.out.println("Zi show");
}
}
- 方法重写(覆盖)的应用:
当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法,这样,即沿袭了父类的功能,又定义了子类特有的内容。
举例:比如手机,当描述一个手机时,它具有发短信,打电话,来电显示功能,后期由于手机需要在来电显示功能中增加显示姓名和头像,这时可以重新定义一个类描述智能手机,并继承原有描述手机的类。并在新定义的类中覆盖来电显示功能,在其中增加显示姓名和头像功能。
在子类中,访问父类中的成员方法格式:
super.父类中的成员方法();
看如下代码:
public class Test {
public static void main(String[] args) {
new NewPhone().showNum();
}
}
//手机类
class Phone{
public void sendMessage(){
System.out.println("发短信");
}
public void call(){
System.out.println("打电话");
}
public void show(){
System.out.println("来电显示号码");
}
}
//智能手机类
class NewPhone extends Phone{
//覆盖父类的来电显示号码功能,并增加自己的显示姓名和图片功能
public void show(){
//调用父类已经存在的功能使用super
super.show();
//增加自己特有显示姓名和图片功能
System.out.println("显示来电姓名");
System.out.println("显示头像");
}
}
-
- 方法重写的必备条件
重写需要注意的细节问题:
- 子类方法覆盖父类方法,必须要保证权限大于等于父类权限。
class Fu(){
void show(){}
public void method(){}
}
class Zi() extends Fu{
public void show(){} //编译运行没问题
void method(){} //编译错误
}
- 写法上稍微注意:必须一模一样:方法的返回值类型 方法名 参数列表都要一样。
总结:当一个类是另一个类中的一种时,可以通过继承,来继承属性与功能。如果父类具备的功能内容需要子类特殊定义时,进行方法重写。
四、重写和重载的区别
/* *1.重载(overload)和重写(override)的区别 * 修饰符 返回值类型 方法名(形参列表){ * * } * public int getSum(int a,int b){ * * } * public double getSum(double a,double b){ * * } * * 重载不一定有继承关系,重写一定有继承关系 * a.修饰符: * 重载对两个方法没有任何要求 * 重写对权限修饰符有要求,子类的方法的权限>=父类的方法的权限 * b.返回值类型: * 重写要求子类的方法和父类中被重写的方法一致 * 重载对两个方法返回值类型没有要求 * c.方法名: * 重写要求子类的方法和父类中被重写的方法方法名一致 * 重载要求两个方法方法名必须一致 * d.形参列表: * 重写要求子类的方法和父类中被重写的方法的形参列表一致 * 重载要求两个方法的形参列表必须不同(形参类型不同,形参个数不同,形参类型顺序不同) */ //重载在继承中应用 //从父类中继承的方法和子类中的方法构成重载关系 class Father{ public int getSum(int a,int b){ return a+b; } }
class Son extends Father{ public double getSum(double a,double b){ return a+b; } public double getSum(double a,double b,double c){ return a+b+c; } } |