Java继承与组合
继承
java 中使用extends关键字表示继承关系,当创建一个类时,如果没有明确指出要继承的类,则是隐式地从根类Object进行继承。
- 子类继承父类的成员变量
- 子类能够继承父类的public和protected成员变量,不能够继承父类的private成员变量
- 如果在子类中出现了同名称的成员变量,则子类的成员变量会屏蔽掉父类的同名成员变量。若要在子类中访问父类中同名成员变量,需要使用super关键字来进行引用。
public class Main { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub Person p = new Student(); System.out.println(p.name); } } class Person { public String name; public String sex; public short age; public Person(){ name = "John"; System.out.println("It is a person"); } } class Student extends Person{ public String name; public Student(){ this.name = super.name; //利用super引用父类变量 System.out.println("It is a student"); } } //输出 It is a person It is a student John
- 子类继承父类的方法
- 能够继承父类的public和protected成员方法,不能够继承父类的private成员方法
- 如果在子类中出现了同名称的成员方法,则称为覆盖,即子类的成员方法会覆盖掉父类的同名成员方法。如果要在子类中访问父类中同名成员方法,需要使用super关键字来进行引用
- 对于父类的静态方法,可以被继承但是不能被子类中的同名方法覆盖;子类中也可以隐藏父类的成员变量,只需要声明的变量名称相同,就算变量的数据类型不同,也一样可以隐藏父类成员变量。
- 构造方法是在新建对象的时候才会执行,静态语句块和静态方法在类加载到内存的时候就已经执行了,当新建一个A类对象时,先调用A的父类的静态方法,再调用A类的静态方法,最后才是A的父类的构造函数、A类的构造函数。
public class Main { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub Person p = new Student(); p.printName(); p.printAge(); } } class Person { private Pet p = new Pet(); /*static{ System.out.println("Person"); }*/ public String name; public String sex; public short age; public Person(String name){ this.name = name; } public void printName(){ System.out.println("My name is" + name); } public static void printAge(){ System.out.println("This is used to print the people's age"); } } class Student extends Person{ public String name; public int age; public Student(){ super("John"); this.name = super.name; //利用super引用父类变量 } public static void printAge(){ System.out.println("This is used to print the student's age"); } public void printName(){ System.out.println("It is used to print the student's name"); } } //输出 It is used to print the student's name This is used to print the people's age
- 构造器
- 子类是不能够继承父类的构造器,但是,如果父类的构造器都是带有参数的,则子类的构造器中必须显示地通过“super”关键字调用父类的构造器并配以适当的参数列表。且必须是子类构造器的第一个语句。如果父类有无参构造器,则在子类的构造器中用super关键字调用父类构造器不是必须的,如果没有使用super关键字,系统会自动调用父类的无参构造器。
class Person { public String name; public Person(String name){ this.name = name; } public void printName(){ System.out.println("My name is" + name); } } class Student extends Person{ public Student(){ super("John"); } public void printName(){ System.out.println("It is used to print the student's name"); } }
组合
组合是指在设计类的同时,把将要组合的类的对象加入到该类中作为自己的成员变量。
class Student extends Person{ public String name; public School school; //组合 public Student(){ super("John"); this.name = super.name; //利用super引用父类变量 } public void printName(){ System.out.println("It is used to print the student's name"); } } class Person { public String name; public Person(String name){ this.name = name; } public void printName(){ System.out.println("My name is" + name); } } class School { public String name; public String address; public void printName(){ System.out.println("School name:" + name); } public void printAddress(){ System.out.println("School address" + address); } }
程序中选择继承或者组合的原则:
- 如果存在一种IS-A的关系(比如Student是一个Person),并且一个类需要向另一个类暴露所有的方法接口,那么更应该用继承的机制。
- 如果存在一种HAS-A的关系(比如Student必有一所School),那么更应该运用组合。