Java抽象类

抽象类介绍

在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。

抽象:不具体,看不明白。抽象类表象体现。

抽象类除了不能实例化对象之外,类的其它功能依然存在,成员变量、成员方法和构造方法的访问方式和普通类一样。

由于抽象类不能实例化对象,所以抽象类必须被继承,才能被使用。也是因为这个原因,通常在设计阶段决定要不要设计抽象类。

父类包含了子类集合的常见的方法,但是由于父类本身是抽象的,所以不能使用这些方法。

 

抽象类的特点:

1)抽象方法只能定义在抽象类中,抽象类和抽象方法必须由abstract关键字修饰(可描述类和方法,不可描述变量)。

2)抽象方法只定义方法声明,并不定义方法实现。

3)抽象类不可以被创建对象(也就是不可以被实例化)。

4)只有通过子类继承抽象类并覆盖了抽象类中的所有抽象方法后,该子类才可以实例化。否则,该子类还是一个抽象类。

 

抽象类的细节:

1)抽象类中是否有构造函数?有,用于给子类对象进行初始化。

2)抽象类中是否可以定义非抽象方法?

可以。其实,抽象类和一般类没有太大的区别,都是在描述事物,只不过抽象类在描述事物时,有些功能不具体。所以抽象类和一般类在定义上,都是需要定义属性和行为的。只不过,比一般类多了抽象方法。而且比一般类少了一个创建对象的部分。

3)抽象关键字abstract和哪些不可以共存?final , private , static

4)抽象类中可不可以不定义抽象方法?可以。抽象方法目的仅仅为了不让该类创建对象。

 

抽象类的定义和使用

在Java语言中使用abstract class来定义抽象类。

abstract class 类名 {
}

使用abstract class来定义抽象类。

下面举个简单的例子:

abstract class Animal{ // 定义一个抽象类
    public void run(){ // 普通方法
        System.out.println("奔跑");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal a = new Animal();  // 无法实例化
    }
}

从上可知,Animal是抽象的,无法直接进行实例化操作。

如果想设计这样一个类,该类包含一个特别的成员方法,该方法的具体实现由它的子类确定,那么你可以在父类中声明该方法为抽象方法。

abstract关键字同样可以用来声明抽象方法,抽象方法只包含一个方法名,而没有方法体。

抽象方法没有定义,方法名后面直接跟一个分号,而不是花括号。

abstract class Animal{ // 定义一个抽象类
    public void run(){ // 普通方法
        System.out.println("奔跑");
    }
    // 抽象方法,没有方法体,使用abstract关键字作修饰
    // 抽象方法所在类一定要标示为抽象类,也就是说该类需要被abstract关键字所修饰。
    public abstract void eat();
}
public class Test {
    public static void main(String[] args) {
        Animal a = new Animal();  // 无法实例化
    }
}

声明抽象方法会造成以下两个结果:

1)如果一个类包含抽象方法,那么该类必须是抽象类。

2)任何子类必须重写父类的抽象方法,或者声明自身为抽象类。

抽象类的使用原则如下: 

1)抽象方法必须为public或者protected(因为如果为private,则不能被子类继承,子类便无法实现该方法),缺省情况下默认为public。

2)抽象类不能直接实例化,需要依靠子类采用向上转型的方式处理。

3)抽象类必须有子类,使用extends继承,一个子类只能继承一个抽象类。 

4)子类(如果不是抽象类)则必须覆写抽象类之中的全部抽象方法(如果子类没有实现父类的抽象方法,则必须将子类也定义为为abstract类)。

 

继承抽象类

继承抽象方法的子类必须重写该方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该抽象方法,否则,从最初的父类到最终的子类都不能用来实例化对象。

//单继承
class Bear extends Animal{ // Bear类是抽象类的子类,是一个普通类
    @Override
    public void eat() { // 强制要求覆写
        System.out.println("吃肉肉!");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal a = new Bear();//向上转型
        a.eat(); // 被子类所覆写的过的方法
    }
}

通过上面的例子我们可以清楚的发现: 

1)抽象类继承子类里面有明确的方法覆写要求,而普通类可以有选择性的来决定是否需要覆写; 

2)抽象类实际上就比普通类多了一些抽象方法而已,其他组成部分和普通类完全一样; 

3)普通类对象可以直接实例化,但抽象类的对象必须经过向上转型之后才可以得到。

虽然一个类的子类可以去继承任意的一个普通类,可是从开发的实际要求来讲,普通类尽量不要去继承另外一个普通类,而是去继承抽象类。

 

抽象类的使用限制

1)抽象类中有构造方法么? 

由于抽象类里会存在一些属性,那么抽象类中一定存在构造方法,其存在目的是为了属性的初始化。 并且子类对象实例化的时候,依然满足先执行父类构造,再执行子类构造的顺序。

// Test.java文件
package com.zwj.test;
abstract class Animal{//定义一个抽象类
    public Animal(){
        System.out.println("*****Animal类构造方法*****");
    }
    public abstract void eat(); // 抽象方法,没有方法体
}
//单继承
class Bear extends Animal{ // Bear类是抽象类Animal的子类,是一个普通类
    public Bear(){
        System.out.println("*****Bear类构造方法*****");
    }
    @Override
    public void eat() { // 强制要求覆写
        System.out.println("吃肉肉!");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal a = new Bear(); // 向上转型
    }
}

我们执行一下看看结果:

*****Animal类构造方法*****
*****Bear类构造方法*****

2)抽象类可以用final声明么? 

不能,因为抽象类必须有子类,而final定义的类不能有子类;

 

3)抽象类能否使用static声明? 

下面是一个关于外部抽象类使用static声明的例子:

// Test.java文件
package com.zwj.test;
static abstract class Animal{ // 定义一个抽象类
    public abstract void eat();
}
class Bear extends Animal{
    public void eat(){
        System.out.println("**********");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal a = new Bear(); // 向上转型
        a.eat();
    }
}

我们来执行一下,看看结果:

Exception in thread "main" java.lang.Error: Unresolved compilation problem: 
    Illegal modifier for the class Animal; only public, abstract & final are permitted

    at com.zwj.test.Animal.<init>(Test.java:2)
    at com.zwj.test.Bear.<init>(Test.java:5)
    at com.zwj.test.Test.main(Test.java:12)

再看一个关于内部抽象类使用static声明的例子:

// Test.java
package com.zwj.test;
abstract class Animal{ // 定义一个抽象类
    static abstract class Bear{ // static定义的内部类属于外部类
        public abstract void eat();
    }
}
class BlackBear extends Animal.Bear{
    public void eat(){
        System.out.println("*****吃肉肉*****");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal.Bear ab = new BlackBear (); // 向上转型
        ab.eat();
    }
}

我们来执行一下:

*****吃肉肉*****

由此可见,外部抽象类不允许使用static声明,而内部的抽象类运行使用static声明。使用static声明的内部抽象类相当于一个外部抽象类,继承的时候使用“外部类.内部类”的形式表示类名称。

4)可以直接调用抽象类中用static声明的方法么? 

任何时候,如果要执行类中的static方法的时候,都可以在没有对象的情况下直接调用,对于抽象类也一样。 

// Test.java
package com.zwj.test;
abstract class Animal{ // 定义一个抽象类
    public static void eat(){
        System.out.println("吃肉肉!");
    }
}
public class Test {
    public static void main(String[] args) {
        Animal.eat();
    }
}

我们试着执行一下,看看结果:

吃肉肉!

 

5)有时候由于抽象类中只需要一个特定的系统子类操作,所以可以忽略掉外部子类。

这样的设计在系统类库中会比较常见,目的是对用户隐藏不需要知道的子类。 

// Test.java
package com.zwj.test;
abstract class Animal { // 定义一个抽象类
    public abstract void eat();
    private static class Bear extends Animal { // 内部抽象类子类
        public void eat(){ // 覆写抽象类的方法
            System.out.println("吃肉肉!");
        }
    }
    //这个方法不受实例化对象的控制
    public static Animal getInstance(){
        return new Bear();
    }

}
public class Test {
    public static void main(String[] args) {
        //此时取得抽象类对象的时候完全不需要知道Bear类这个子类的存在
        Animal a = Animal.getInstance();
        a.eat();
    }
}

我们来看下执行结果:

吃肉肉!

 

模板方法设计模式

解决的问题:当功能内部一部分实现是确定的,一部分实现是不确定的。这时可以把不确定的部分暴露出去,让子类去实现。

abstract class Demo {
    public final void getTime(){  // 此功能如果不需要复写,可加final限定
        long start = System.currentTimeMillis();
        code();  // 不确定的功能部分,提取出来,通过抽象方法实现
        long end = System.currentTimeMillis();
        System.out.println("毫秒是:" + (end - start));
    }
    public abstract void code();  // 抽象不确定的功能,让子类复写实现
}
class SubDemo extends Demo {
    public void code(){  // 子类复写功能方法
        for(int i = 0; i < 1000; i++){
            System.out.println("i");
        }
    }
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值