目录
一、继承
1.定义
继承是一种设计思想,它可以从已有的类里面派生出新的类。新的类拥有已有的类的属性和行为。并且可以拓展新的功能。
类B继承类A,则B称为子类,A称为父类。
2.为什么(何时)使用继承?
逻辑上满足“is-a”关系。如布偶是一只猫,猫是一种动物。
如同将对象进行分类一样,同样也可以对类进行分类。将子类共有的属性和行为存放到父类中。
3.具体实现
使用extends关键字实现继承。格式为:
[访问权限修饰符] [修饰符] 子类名 extends 父类名{ 子类体 }
(1)传递性
继承具有传递性。若有ABC三个类,类B是类A的子类,类C是类B的子类。那么类C除了继承了类B的属性和方法之外,还通过类B去间接继承了类A对的属性和方法。
当一个类没有继承任何一个类的时候,jvm会默认让该类继承Object类。Object类是所有类的基类。
(2)单继承
java中一个子类只能继承一个直接父类,但是一个父类可以被多个子类继承。
(3)代码复用性
将子类共有的属性和行为存放到父类中。
实际上和之前方法中提到的某一段代码被频繁重复使用,就可以将其定义成一个方法的思路一样。
比如男人和女人都有眼睛(属性),都会吃饭(行为/方法);假如不使用继承,那么在这两个类中就要重复编写它们,但是使用继承后,就可以提高复用性,还有利于代码维护。
(4)代码扩展性
子类不仅继承了父类的属性和行为,还可以在类中添加自己特有的功能且不影响其他类。
比如父类是人类,子类是中国人和外国人。在中国人类里面可以添加使用筷子的方法实现代码扩展。
二、继承中的构造方法和关键字Super
1.继承中的构造方法
继承后,子类的构造方法会先调用父类的构造方法。
如图:创建父类Human,并将无参构造方法显示出来。
public class Human {
public Human() {
System.out.println("父类构造方法");
}
}
创建子类Chinese继承Human,并将无参构造方法显示出来。 并在主方法中创建一个Chinese对象。
public class Chinese extends Human {
public Chinese(){
System.out.println("子类构造方法");
}
public static void main(String[] args) {
new Chinese();
}
}
运行结果如下:
2.关键字super
super可以访问父类成员。
(1)super()和super(参数列表)
都是用来调用父类的构造方法。super()用来调用父类的无参构造方法,super(参数列表)用来定义父类的有参构造方法。必须在子类构造方法的第一行。在没有显示使用时,默认调用父类无参构造方法。
如:如图所示,上个示例的类Human不变,类Chinese内改为有参构造方法并在main里使用有参构造方法创建对象。
public class Chinese extends Human {
//有参构造方法
public Chinese(String name,String gender){
System.out.println("子类有参构造方法");
}
//调用子类有参构造方法。
public static void main(String[] args) {
new Chinese("a","b");
}
}
运行结果:
同样地,在子类的无参和有参构造方法里,都可以调用父类的有参构造方法,但需要显示调用。 有一个小细节就是:需要注意代码的执行顺序。
如将上述示例父类Human改为如下所示
public class Human {
//有参构造方法
public Human(String name,String gender) {
System.out.println("父类构造方法");
}
}
再将Chinese改为如下所示
public class Chinese extends Human {
//无参构造方法
public Chinese(){
//调用父类有参构造方法
super("c","y");
System.out.println("子类构造方法");
}
public static void main(String[] args) {
new Chinese();
}
}
这样就可以调用,运行结果为:
但是如果对类Chinese稍作修改 :
public class Chinese extends Human {
//设置两个成员变量并赋值
String a="c";
String g="y";
public Chinese(){
//调用父类有参构造方法
super(a,g);
System.out.println("子类构造方法");
}
public static void main(String[] args) {
new Chinese();
}
}
此时就会报错,原因是看似代码中好像给变量a和g已经赋值了,事实上在类加载中只会声明这两个变量,随后在创建对象即调用构造方法时,才会进行内存分配和变量赋值。但是在这之前首先要调用父类的有参构造方法,由于此时要使用两个变量,而变量还未赋值,所以就会报错。
(2)super.成员变量名(或者方法名)
实现调用父类的成员变量或者成员方法。要注意父类成员的访问权限。
三、方法重写
1.定义
当父类的方法功能无法满足子类的需求的时候,就需要用到方法重写。
如人类会说话,所以在父类Human中定义一个说话的方法,但是中国人说汉语,美国人说英语。这时候就需要在各自的类里面进行方法重写。
2.具体实现
方法重写要求子类的方法结构和父类的方法结构(即方法名、返回值类型、参数列表)须一致,而子类的访问权限大于等于父类的访问权限即可。
注意:构造方法和静态方法不能重写。 构造方法要求方法名与类名相同,但是方法重写要求子类和父类方法结构一致,这两点相冲突。 而子类和父类的静态方法可以写的一模一样,但是实际上因为是通过类名调用,他们都是与自己的类名直接绑定,通过类名确定的是调用的是哪个静态方法。所以只是子类隐藏了父类的静态方法。
3.标签@override
用来明确标识进行重写的方法,在需要重写的方法前插入。可以验证方法重写的正确性,同时也增加了代码的可读性和维护性。
四、抽象类和抽象方法
1.抽象方法
在经过本篇前面的内容学习后,会发现,越是顶层的类,越没有必要去实现具体细节,也就越抽象。例如人类会说话,但是各国人语言不同,所以只需要定义这个方法即可,不用实现方法的细节。
所以抽象方法就是只有声明,而没有具体的实现的方法。
2.关键字abstract
用来修饰抽象的类或方法。
格式为: [访问权限修饰符] abstract [返回值类型] 方法名; 注意没有{ },即方法体。
3.抽象类
当一个类中没有足够的信息可以描绘出一个具体的对象,即为抽象类。用abstract修饰的类就是抽象类。包含抽象方法的类一定需要用abstract修饰。
抽象类不能实例化对象。它只能作为一种基类被继承,继承它的类也可以是抽象类,倘若不是一个抽象类,就要利用方法重写实现所有抽象方法的具体细节。