1.抽象类(abstract class):使用了abstract关键字所修饰的类叫做抽象类。抽象类无法实例化,也就是说,不能new出来一个抽象类的对象(实例)。
抽象类的定义格式:
public abstract class abstractTest
{
}
public class Test
{
public static void main(String[] args)
{
T t = new T();
}
}
abstract class T
{
}
执行结果:
D:\src>javac Test.java
Test.java:5: 错误: T是抽象的; 无法实例化
T t = new T();
^
1 个错误
【说明】:以上程序错误的原因是抽象类不能被实例化
2.抽象方法(abstract method):使用abstract关键字所修饰的方法叫做抽象方法(有声明,无实现)。抽象方法需要定义在抽象类中。相对于抽象方法,之前所定义的方法叫做具体方法(有声明,有实现)。
public class Test
{
public static void main(String[] args)
{
}
}
abstract class T
{
public abstract void method();
}
上例程序中定义一个抽象方法:public abstract void method();注意直接用分号结束,没有花括号,没有花括号表示没有实现这个方法,注意的是如果有花括号即便是花括号里面没有内容,这也算是有实现的,只不过这个方法体为空,空实现。
3.如果一个类包含了抽象方法,那么这个类一定是抽象类。
4.如果某个类是抽象类,那么该类可以包含具体方法(有声明,有实现)。
public class Test
{
public static void main(String[] args)
{
}
}
abstract class T
{
public abstract void method();
public void test()
{
System.out.println("test");
}
}
编译成功!
5.如果一个类中包含了抽象方法,那么这个类一定要声明成abstract class,也就是说,该类一定是抽象类;反之,如果某个类是抽象类,那么该类既可以包含抽象方法,也可以包含具体方法,甚至什么都不包含也是可以的。
6.无论何种情况,只要一个类是抽象类,那么这个类就无法实例化。
7.在子类继承父类(父类是个抽象类)的情况下,那么该子类必须要实现父类中所定义的所有抽象方法;如果子类不实现父类的所有方法,否则,该子类需要声明成一个abstract class。
public class Test
{
public static void main(String[] args)
{
R r = new R();
}
}
abstract class T
{
public abstract void method();
public void test()
{
System.out.println("test");
}
}
class R extends T
{
}
编译结果:
D:\src>javac Test.java
Test.java:18: 错误: R不是抽象的, 并且未覆盖T中的抽象方法method()
class R extends T
^
1 个错误
两种解决办法:
第一种实现父类的抽象方法:
public class Test
{
public static void main(String[] args)
{
R r = new R();
}
}
abstract class T
{
public abstract void method();
public void test()
{
System.out.println("test");
}
}
class R extends T
{
public void method()
{
System.out.println("method");
}
}
编译通过!
第二种,将子类转为抽象类
public class Test
{
public static void main(String[] args)
{
// R r = new R();
}
}
abstract class T
{
public abstract void method();
public void test()
{
System.out.println("test");
}
}
abstract class R extends T
{
}
编译通过!
【说明】:此时子类还是不能被实例化,如果想使用这个子类,还必须定义一个R的一个子类,把method的方法实现出来。
继续修改以上例子:
public class Test
{
public static void main(String[] args)
{
// R r = new R();
}
}
abstract class T
{
public abstract void method();
public void test()
{
System.out.println("test");
}
}
abstract class R extends T
{
public void method()
{
System.out.println("method");
}
}
编译成功!
【说明】:R继承了T,则R类里面有两个方法,一个方法是实现T的抽象方法,另外一个是从T当中继承下来的具体方法test();此时把R声明为抽象类正常,应该抽象类本身可以包含具体方法,也可以包含抽象方法。
8.抽象方法,抽象类有什么作用,有声明没实现?
这个实现是推迟到子类当中去实现的,父类实现不了,父类可以声明一个抽象方法,写一个注释,告诉这个类的实现者,告诉它我这个方法有什么用,可以让子类去实现,对于一个父类来说,可以有多个子类,而不同的子类可以实现不同的方法。比如说在父类中定义一个排序的抽象方法,而在这个父类的各个子类当中可以分别实现这个抽象方法,具体子类实现方法有可能是用冒泡排序,有可能是用哈希排序,每种实现规则的不一样,从这一点可以看出,抽象类和抽象方法主要起到一个规范,约束的作用。
举个例子加深理解
public class Test2
{
public static void main(String[] args)
{
Shape shape = new Triangle(10,6);
int area = shape.computerArea();
System.out.println("triangle:" + area);
shape = new Rectangle(10,10); //前面area用完,shape指向完已经关闭,下面可以不用定义直接使用。
area = shape.computerArea();
System.out.println("rectangle" + area);
}
}
abstract class Shape
{
public abstract int computerArea(); //计算形状面积
}
class Triangle extends Shape
{
int width;
int height;
public Triangle(int width, int height) //用构造方法来完成一些初始化工作
{
//注意此处用this是因为生命完构造方法之后,把方法外面的int width;和int height;给屏蔽掉了,所以这个地方要访问到外面的width和height成员变量就用this,this表示对自己的引用。这样就把方法参数传给成员变量。
this.width = width;
this.height = height;
}
public int computerArea()
{
return width * height / 2;
}
}
class Rectangle extends Shape
{
int width;
int height;
public Rectangle(int width,int height)
{
this.width = width;
this.height = height;
}
public int computerArea()
{
return width * height;
}
}
执行结果:
D:\src>java Test2
triangle:30
rectangle100
【说明】:从这个例子说可以看到抽象方法起到规范,规则子类行为的作用具体怎么实现满足这个规则由子类去实现。约束规则,比如说定义什么类型,传递什么参数等。