- 👑专栏内容:Java
- ⛪个人主页:子夜的星的主页
- 💕座右铭:前路未远,步履不停
目录
- 一、抽象类的概念
- 二、注意事项
- 三、抽象类的作用
一、抽象类的概念
在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。
在Java中被 abstract
修饰称为抽象类,抽象类中被 abstract
修饰的方法称为抽象方法,抽象方法不用给出具体的实现体。要注意:抽象类也是类,内部可以包含普通方法和属性,甚至构造方法。抽象类在编程中的作用就是创建一个抽象类通常是为了提供一个基础的、可以被其他更具体的类继承和扩展的框架。
举个实际的例子,以下面的图形类为例。
class Shape{
public void draw(){
//System.out.println("画图");
}
}
class Cycle extends Shape{
@Override
public void draw() {
System.out.println("⭕");
}
}
class Rect extends Shape{
public void draw(){
System.out.println("矩形");
}
}
public class demo1 {
public static void drawMap(Shape shape){
shape.draw();
}
public static void main(String[] args) {
drawMap(new Rect());
drawMap(new Cycle());
}
}
在这个打印图形例子中发现,父类 Shape 中的draw
方法好像并没有什么实际工作。即使父类的draw方法里面不写任何内容,也不影响程序的正常进行。主要的绘制图形都是由 Shape
的各种子类的 draw
方法来完成的。像这种没有实际工作的方法,我们可以把它设计成一个抽象方法,包含抽象方法的类我们称为抽象类。 现在可以将上面的 Sharp
类更改为抽象类。
abstract class Sharp{
public abstract void draw ();
}
二、注意事项
-
抽象类和抽象方法的修饰符: 在编程中,抽象类和抽象方法都是使用关键字
abstract
进行修饰的。这个修饰符标识着类或方法是抽象的,即它们不完全具体化,需要在子类中进一步定义或实现。 -
抽象类的实例化: 抽象类不能被实例化,这意味着你不能直接创建一个抽象类的对象。这是因为抽象类通常包含不完整的方法,需要子类来具体实现这些方法。
-
抽象方法和抽象类的关系: 一个类如果包含至少一个抽象方法,则这个类必须被声明为抽象类。但是,一个抽象类不一定要包含抽象方法,它也可以只提供一些通用的实现。
-
抽象类中的成员: 抽象类可以包含普通的成员变量和成员方法。这些成员变量和方法可以被子类继承和使用,或者在子类中被覆盖。
-
继承抽象类的普通类: 当一个普通类继承一个抽象类时,它必须实现(或称为“重写”)抽象类中的所有抽象方法。这是因为普通类必须是完全具体化的。
-
抽象类的意义: 抽象类的主要目的是为了被其他类继承。它提供了一个共同的模板,确保所有子类都具有一定的通用行为。
-
抽象类继承与方法重写: 当一个抽象类继承另一个抽象类时,它可以不立即实现父类中的抽象方法。但是,如果一个普通类继承一个抽象类,那么它必须实现所有继承的抽象方法。
-
抽象类的限制: 由于抽象类的目的是为了被继承和实现,它们不能被声明为
final
或static
,这些修饰符会限制类的继承和方法的覆盖。抽象方法也不能是private
,因为私有方法不能被子类访问或重写。 -
final
与abstract
的冲突:final
关键字表示一个类或方法不能被修改或继承,而abstract
表示需要进一步的实现或扩展。因此,这两个关键字在概念上是对立的,不能同时用于同一个类或方法上。 -
抽象类中的构造方法: 抽象类可以有构造方法。这些构造方法不用于直接创建对象,而是在子类对象的创建过程中用于初始化继承自抽象类的成员变量。
三、抽象类的作用
抽象类本身不能被实例化,要想使用,只能创建该抽象类的子类。然后让子类重写抽象类中的抽象方法。
【问题】普通的类也可以被继承,普通的方法也可以被重写,为啥非得用抽象类和抽象方法呢?
使用抽象类相当于多了一重编译器的校验。使用抽象类的场景就如上面的代码,实际工作不应该由父类完成, 而应由子类完成。那么此时如果不小心误用成父类使用普通类编译器是不会报错的。但是父类是抽象类就会在实例化的时候提示错误,让我们尽早发现问题。
很多语法存在的意义都是为了 “预防出错”,例如final
也是类似。 创建的变量用户不去修改,不就相当于常量嘛?但是加上final
能够在不小心误修改的时候,让编译器及时提醒我们。