72.抽象类
在我们编写一个对象的时候,我们可能会对一些类的方法进行定义,但是并不具体实现。而是将这些方法的实现放到它的子类中去。这样可以增强类设计的灵活性。
比如,我们定义了一个表示各种图形的类Shape,这个类有一些属性,还有一个用于计算这个图形的周长的方法calPerimeter(),但是,对于不同的图形,对周长的计算方法也不同,我们不能将所有的图形的周长的计算方法都写到这个方法中。
通过关键字abstract,我们可以在父类Shape中不实现这个方法,而将它的实现放到子类中去。比如,一个圆形的类继承了Shape这个父类,然后,在这个子类中实现calPerimeter()这个方法的细节:圆形的周长=2ПR,其中, П表示圆周率,而R表示圆的半径。如果是一个矩形继承了这个类,则可以在这个子类中实现这个方法calPerimeter(),它的计算方式是2*(Width+Heigt)。
从上面的讨论中可以看出来,我们有时候只是定义某个类的一个“骨架”,并不具体实现,而将进一步的具体实现放到子类中去完成。这个时候,我们可以将这个类定义为abstract的,而将没有实现的方法声明为abstract的。因为abstract类中具有没有被实现的方法,所以,抽象类不能被实例化。
这样做,比起在类里面留一个未实现的空方法来说,要安全的得多,会防止其他人误用,导致难以发现问题的bug。使用抽象类,会让使用该类的人明确知道,需要由自己来根据实际情况来实现这个类中的部分方法。
抽象类里面并非一定需要抽象方法,当你定义了一个类,但是又不想让它被直接实例化的时候,可以使用抽象类的方法来实现。反之,如果类中有抽象方法,则一定要将类定义为抽象的类。
在以下任一条件成立时,类必须定义成抽象类:
类中有至少一个抽象方法;
类继承了父类中的抽象方法,但是至少有一个抽象方法没有实现;
类实现了某个接口,但没有全部实现接口中的方法。
下面我们来看一个抽象类的例子。
首先我们定义了一个用来表示各种图形的类Shapes,这个类有两个抽象方法,一个用于返回图形的形状,一个用于返回图形的周长:
public abstract class Shapes {
private String color;
/**
* 得出周长
*/
public abstract double perimeter();
/**
* 得到形状
*/
public abstract String getType();
// 用于设置/获取“颜色”属性的方法
public String getColor() {
return color;
}
public void setColor(String theColor) {
color = theColor;
}
}
注意,这是一个抽象类,不能直接实例化它。
public class ShapeTriangle extends Shapes {
protected double a, b, c;
public ShapeTriangle() {
setSides(0.0, 0.0, 0.0);
}
public ShapeTriangle(double i, double j, double h) {
setSides(i, j, h);
}
public void setSides(double x, double y, double z) {
this.a = x;
this.b = y;
this.c = z;
}
/**
* 实现父类中的抽象方法
*/
public double perimeter() {
return a + b + c;
}
public String getType() {
return "三角形";
}
public static void main(String args[]) {
ShapeTriangle st = new ShapeTriangle(3.0, 4.0, 5.0);
System.out.println("形状:" + st.getType());
System.out.println("周长:" + st.perimeter());
}
}
这是一个表示三角形的类,这个类继承了父类Shapes,然后,在父类的基础上新增了一些属性以及相应的方法。
根据三角形的实际情况,实现了父类中的两个抽象方法:getType()返回一个“三角形”的字符串,而且根据三角形周长的算法实现了父类中的计算周长的方法:perimeter。
而对于圆形,可能需要这样来实现它的两个抽象方法:
public class ShapeCircle extends Shapes {
private double r;
public ShapeCircle() {
setRadius(0.0);
}
public ShapeCircle(double ra) {
setRadius(ra);
}
public void setRadius(double rad) {
this.r = rad;
}
/**
* 实现父类的抽象方法
*/
public double perimeter() {
return 2.0 * Math.PI * r;
}
public String getType() {
return "圆";
}
public static void main(String args[]) {
ShapeCircle sc = new ShapeCircle(5);
System.out.println("形状:" + sc.getType());
System.out.println("周长:" + sc.perimeter());
}
}
在这个类ShapeCircle中,也继承了类Shapes,但它对抽象方法perimeter()的实现方式是和ShapeTriangle不一样的。
注意:
区分没有实现的方法和空方法体的方法的区别,下面这个方法是没有实现的方法:public int methodA();
而下面这个方法是空方法体的方法:public int methodB(){}
没有实现的方法可以用abstract来修饰,而空方法体方法却不能使用abstract来修饰,它已经实现了方法,只是在这个方法中什么动作也没有做而已。
上一篇: matlab-矩阵合并
下一篇: 2.Java技术体系架构