抽象类(一):
为何会出现抽象类这种类?
1、某些情况下,父类只知道子类需要哪些方法,但是不知道子类如何实现这些方法,或者每个子类对同一方法的实现都不同
2、所以允许父类定义方法时,只定义方法,但是不具体实现,定义的这种方法称为抽象方法,包含这种抽象方法的类,就被称为抽象类
3、抽象方法:只有方法签名,没有方法实现的方法,抽象方法用 abstract 修饰 ,如:
public abstract double getHeight();
4、上面的代码定义了一个 抽象方法:getHeight();只有方法签名,没有后面的方法体。
因此总结出抽象方法和抽象类的一些规则如下:
1、抽象类和抽象方法必须用 abstract 修饰符来修饰
2、抽象方法不能有方法体
3、抽象类不能被实例化,无法使用 new 关键字来调用抽象类的构造器来实现实例对象,即使抽象类里不包含抽象方法,这个类也不能被实例化
4、抽象类可以包含成员变量、方法(普通方法、抽象方法都可以)、构造器,初始化块,内部类(接口、枚举)5种成分
5、抽象类的构造器不能用于创建实例,只能用于被其子类调用
6、包含抽象方法的类,只能被定义为抽象类
7、抽象类里可以没有抽象方法
8、类包含抽象方法是指:
.8.1、直接定义了一个抽象方法
.8.2、继承了一个抽象父类,但没有完全实现父类的抽象方法
.8.3、实现了一个接口,但没有完全实现接口包含的抽象方法
9、归纳起来,抽象类可以用 “有得有失” 四个字来形容:
有得:抽象类多了一个能力,能包含抽象方法
有失:抽象类失去了一个能力,不能用于创建实例
10、使用 abstract 修饰的类必须被继承,使用 abstract 修饰的方法必须被重写
使用 final 修饰的类不能被继承,使用 final 修饰的方法不能被重写
所以 abstract 和 final 永远不能一起用
如下代码定义了一个抽象类,并被不同的子类继承实现:
import static java.lang.System.*; //-定义一个抽象类,作为父类 abstract class Shape{ { out.println("执行Shape类的普通初始化块!"); } private String color; //-定义两个抽象方法,让子类去重写实现 public abstract double getPerimeter(); public abstract String getType(); public Shape(){ } public Shape(String color){ out.println("执行带参数的Shape类构造器!"); this.color=color; } public String getColor(){ return this.color; } public void setColor(String color){ this.color=color; } } //-子类 继承抽象父类 class Triangle extends Shape{ private double a; private double b; private double c; public Triangle(String color,double a,double b,double c){ //-super关键字 调用父类构造器 super(color); setSides(a,b,c); } public void setSides(double a,double b,double c){ if(a>=b+c||b>=a+c||c>=a+b){ out.println("三角形的两边之和必须大于第三边!"); return; } this.a=a; this.b=b; this.c=c; } //-重写实现父类 getPerimeter() 方法 public double getPerimeter(){ return a+b+c; } //-重写实现父类 getType() 方法 public String getType(){ return "三角形"; } } //-子类 继承抽象父类 public class Circle extends Shape{ private double radius; public Circle(String color,double radius){ //-super关键字 调用父类构造器 super(color); this.radius=radius; } //-重写实现父类 getPerimeter() 方法 public double getPerimeter(){ return 2*Math.PI*radius; } //-重写实现父类 getType() 方法 public String getType(){ return "圆形"; } public void setRadius(double radius){ this.radius=radius; } public double getRadius(){ return this.radius; } public static void main(String[] args){ //-分别调用两个子类中重写的 方法: Shape s1=new Triangle("红色",3,4,5); out.println("周长是:"+s1.getPerimeter()); out.println("图形是:"+s1.getType()); out.println(); Shape s2=new Circle("蓝色",5); out.println("周长是:"+s2.getPerimeter()); out.println("图形是:"+s2.getType()); out.println(); //-改变一下写法: Triangle s3=new Triangle("红色",3,4,5); out.println("周长是:"+s3.getPerimeter()); out.println("图形是:"+s3.getType()); out.println(); Circle s4=new Circle("蓝色",5); out.println("周长是:"+s4.getPerimeter()); out.println("图形是:"+s4.getType()); } }
运行结果:
如上代码看到:利用抽象类和抽象方法,可以更好的发挥多态的优势,使程序可以更灵活,但是不用抽象类抽象方法,结果是一样的。
注:
1、abstract 不能修饰成员变量,不能修饰局部变量,即:没有抽象成员变量、抽象局部变量的说法
2、abstract 不能修饰构造器,也没有抽象构造器的说法,抽象类里定义的构造器只能是普通构造器
3、当使用 static 修饰方法时,表示这个方法属于类本身,即通过类就能调用该方法,但如果该方法被定义为抽象方法,当用该类来调用该方法时,就会出现错误,即调用了一个没有方法体的方法,因此: static 和 abstract 不能同时修饰一个方法,即没有所谓的类抽象方法
4、static 和 abstract 并不是绝对互斥的,他们可以同时修饰内部类
5、abstract 修饰的方法只有被子类重写才有意义,因此 abstract 方法不能定义为 private 访问权限,即:abstract 和 private 不能同时修饰方法