继承 :子类继承父类,子类将得到父类的全部方法和Field,但不能获得父类的构造器,一个子类只有一个直接的父类,用 extends 关键字来实现。
1、子类将得到父类的全部方法 ?。这得分三种情况讨论:
a、父类方法用private修饰,子类对象将无法访问该方法。
b、父类方法与子类方法同名(方法名、形参名),子类方法将覆盖父类方法(或者叫重写),子类对象将无法直接访问该方法。但可以间接访问,这里就要用到super关键字或父类名来访问了,具体下面再讨论。
c、除去上面2种情况,子类对象才能直接访问父类方法。
1 class BaseClass 2 { 3 private void test(){ 4 System.out.println("父类test方法"); 5 } 6 public void base(){ 7 System.out.println("父类base方法"); 8 } 9 void kass(){ 10 System.out.println("父类的kass方法"); 11 } 12 } 13 public class SubClass extends BaseClass 14 { 15 public void kass(){ 16 System.out.println("子类的kass方法"); 17 } 18 public static void main(String[] args){ 19 SubClass bc = new SubClass(); 20 //bc.test(); //因父类test方法设值了private权限,故这里无法访问。 21 bc.kass(); //方法相同,子类覆盖父类方法,这里只能访问子类方法。 22 bc.base(); // 正常访问父类base方法。 23 } 24 }
这里还要就子类重写父类方法说明几点:
1、方法名相同、形参列表相同。如果子、父类都用private修饰符,子类依旧不能访问父类方法。
2、子类方法返回值类型应比父类更小或相等,子类方法声明抛出的异常类应比父类更小或相等。
3、子类方法的访问权限应比父类更大或相等。权限从小到大:省略修饰符<protected<public 。
4、方法覆盖的子、父类方法要么都是类方法,要么都是实例方法。也就是要么都用static修饰,要么都不用。
关于子类对象调用父类被覆盖的方法:
1、类方法:通过 父类类名.方法名(形参列表); 来访问。
2、实例方法:通过 super.方法名(形参列表);来访问。
说明:只能在子类方法中通过父类类名或super来调用父类被覆盖的方法。
1 class BaseClass 2 { 3 static void test(){ 4 System.out.println("父类test方法"); 5 } 6 public void kass(){ 7 System.out.println("父类的kass方法"); 8 } 9 } 10 public class SubClass extends BaseClass 11 { 12 static void test(){ 13 System.out.println("子类test方法"); 14 BaseClass.test(); //调用父类test方法 15 } 16 public void kass(){ 17 System.out.println("子类的kass方法"); 18 super.kass(); //调用父类kass方法 19 } 20 public static void main(String[] args){ 21 SubClass bc = new SubClass(); 22 bc.test(); 23 bc.kass(); 24 } 25 }
2、子类将得到父类的全部Field?这也得分三种情况讨论:
a、父类Field用private修饰,子类对象将无法访问该Field。
b、父类Field与子类Field同名,子类Field将隐藏父类Field,子类对象将无法直接访问父类Field。但可以间接访问,这里就要用到super关键字或父类名来访问了,具体下面再讨论。为什么叫隐藏,因为子类创建对象时依然为父类Feild分配内存空间。
c、除去上面2种情况,子类对象才能直接访问父类Field。
1 class BaseClass 2 { 3 static int a = 5; 4 public int b = 8; 5 private int c = 11; 6 } 7 public class SubClass extends BaseClass 8 { 9 static int a = 5; 10 public int b = 8; 11 private int c = 11; 12 public void test(){ 13 System.out.println(super.b);//super只能放在非静态方法中 14 System.out.println(super.c);//子类中无法访问父类private修饰的Field,故会报错 15 } 16 public static void main(String[] args){ 17 SubClass bc = new SubClass(); 18 System.out.println(BaseClass.a); //静态修饰的Field用父类名.Field名 调用 19 bc.test(); 20 } 21 }
说明:如果在子类某个方法中访问名为a的Feild,但没有super或父类名调用,则系统查找a的顺序是:
1、查到当前方法是否有名为a的局部变量。
2、查找当前类是否有名为a的Feild。
3、查找父类中是否包含名为a的Feild,依次上溯a的所有父类,直到java.lang.Object类,最终找不到将出现编译错误。
3、不能获得父类的构造器,但可以调用父类构造器的初始化代码
在一个构造器调用另一个重载的构造器使用this调用来完成,在子类构造器中调用父类构造器使用super调用来完成,super调用必须在构造器代码首行。
不管是否用super来调用父类构造器初始化代码,子类构造器总会调用父类构造器一次,如果父类构造器是无参构造器,将不会输出任何信息。
创建任何对象总是从该类所在继承树最顶层类的构造器开始执行,然后依次向下执行,最后才到本类构造器。
1 class BaseClass 2 { 3 public int a; 4 public String b; 5 public BaseClass(int a,String b){ 6 this.a = a; 7 this.b = b; 8 } 9 } 10 public class SubClass extends BaseClass 11 { 12 public double c; 13 public SubClass(int a,String b,double c){ 14 super(a,b); //通过super调用父类构造器初始化过程 15 this.c = c; //这里是this引用 16 } 17 public static void main(String[] args){ 18 SubClass bc = new SubClass(12,"哈喽",32.21); 19 System.out.println(bc.a+" "+bc.b+" "+bc.c); 20 } 21 }
4、继承的硬伤,破坏封装
1、继承是实现类复用的重要手段,但破坏了封装,而采用组合方式实现类复用则提供更好的封装性。
2、子类如果继承及父类的全部方法和Field,那么父类将完全暴露在子类下,如何不让子类随意修改父类:
1、尽量隐藏父类的内部数据,将成员变量设置成private访问类型。
2、不要让子类随意访问、修改父类方法,用private修饰父类方法限制子类访问,用final修饰父类方法防止子类重写,用protected父类方法限制只能被子类重写。
3、尽量不要在父类构造器中调用将要被子类重写的方法。