面向过程的范式重点在于方法的设计,而面向对象的范式将数据和方法结合在对象中。Java在软件重用方面有一个重要且功能强大的特征。假设要定义一个类,对圆、矩形和三角形建模。这些类有很多共同的特性。设计这些类来避免冗余并使系统更易于理解和维护的最好方式是什么?答案就是使用继承。
父类和子类
继承使得你可以定义一个通用的类(即父类),之后扩充该类为一个更加特定的类(即子类)。不同类也可能会有一些共同的特征和行为,这些共同的特征和行为都统一放在一个类,它是可以被其他类所共享的。这些特定的类继承通用类中的特征和方法。子类从它的父类中继承可访问的数据域和方法,还可以添加新数据域和新方法。
package Inherit;
import java.util.Date;
public class SimpleGeometricObject {
private String color = "white";
private boolean filled;
private java.util.Date dateCreated;
public SimpleGeometricObject(){
dateCreated = new java.util.Date();
}
public SimpleGeometricObject(String color, boolean filled){
dateCreated = new java.util.Date();
}
public String getColor(){
return color;
}
public void setColor(String color){
this.color = color;
}
public boolean isFilled(){
return filled;
}
public void setFilled(boolean filled){
this.filled = filled;
}
public java.util.Date getDateCreated(){
return dateCreated;
}
public String toString(){
return "created on" + dateCreated + "\ncolor:" + color + " and filled: "+ filled;
}
}
package Inherit;
public class CircleFromSimpleGeometricObject extends SimpleGeometricObject{
private double radius;
public CircleFromSimpleGeometricObject(){
}
public CircleFromSimpleGeometricObject(double radius){
this.radius = radius;
}
public CircleFromSimpleGeometricObject(double radius, String color, boolean filled){
this.radius = radius;
setColor(color);
setFilled(filled);
}
public double getRadius(){
return radius;
}
public void setRadius(double radius){
this.radius = radius;
}
public double getArea(){
return radius * radius * Math.PI;
}
public double getPerimeter(){
return 2 * radius * Math.PI;
}
public void printCircle(){
System.out.println("The circle id created " + getDateCreated() + " and the radius is " + radius);
}
}
关键字extends告诉编译器,Circle类扩展自GemetricObject类,这样,它就继承了getColor、setColor、isFilled、setFilled和toString方法。
重载的构造方法CircleFromSimpleGeometricObject(double radius, String color, boolean filled)是通过setColor和setFilled方法设置color和filled属性来执行的。这两个公共方法是在父类中定义的,并在Circle中继承,因此可以在Circle类中使用它们。
你可能会尝试在构造方法中使用数据域color和filled。
public CircleFromSimpleGeometricObject(double radius, String color, boolean filled){
this.radius = radius;
this.color = color;
this.filled = filled;
}
这是错误的,因为父类中的私有数据域color和filled是不能被除了父类本身之外的其他任何类访问的。唯一读取和改变color与filled的方法就是通过它们的get和set方法。
与传统的理解不同,子类并不是父类的一个子集。实际上,一个子类通常比它的父类包含更多的信息和方法。一个Java类只可能直接继承自一个父类,这种限制称为单一继承。
使用super关键字
关键super指代父类,可以用于调用父类中的普通方法和构造方法。
(1)调用父类的构造方法
父类的构造方法不会被子类继承。它们只能使用关键字super从子类的构造方法中调用。
语法:super() 或者 super(parameters)
语句super()调用父类的无参构造方法,而语句super(parameters)调用与参数匹配的父类的构造方法。语句super() 和 super(parameters)必须出现在子类构造方法的第一行,这是显式调用父类构造方法的唯一方式。
例如:
public CircleFromSimpleGeometricObject(double radius, String color, boolean filled){
super(color,filled);
this.radius = radius;
}
构造方法链
构造方法可以调用重载的构造方法或父类的构造方法。如果它们都没有被显式地调用,编译器就会自动地将super()作为构造方法的第一条语句
public ClassName(){ public ClassNmae(){
//some statements 等价于 super();
//some statementse
} }
(2)调用父类的方法
方法的重写
方法重写与重载
Object类及其toString()方法