项目代码
https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter10/src/com/yinhai/abstract_
一、抽象类的引入
很多时候在创建类的时候有一个父类,比如animal类,他的子类会有各种方法,为了复用需要进行方法的重写,比如子类Cat会eat(),所以不得已只能在animal类内也需要写eat()方法,但没有意义。
所以当父类的某些方法,需要声明,但是又不确定如何实现时,可以将其声明为抽象方法,那么这个类就是抽象类。
abstract class Animal{
    // public void eat(){System.out.println("eat但不知道谁在吃");}//所以需要抽象方法
        //这里实现了eat(),但其实没有什么意义
        //即 父类方法不确定性的问题
        //===>考虑该方法设计为抽象(abstract)方法
        //==>所谓抽象方法就是没有实现的方法
        //===>所谓没有实现就是指,没有方法体
    abstract public void eat();
        //注意 当一个类中存在抽家方法时,需要将该类声明为abstract类
        //所以一般是子类来实现这个抽象方法的重写
}
class Cat extends Animal{
    public void eat(){
        System.out.println("小猫在吃....");
    }
} 
二、抽象类的基本介绍
1)用abstract关键字来修饰个类时,这个类就叫抽象类访问修饰符abstract类名{
2)用abstract关键字来修饰一个方法时,这个方法就是抽象方法访问修饰符abstract返回类型方法名(参数列表)//没有方法体

3)抽象类的价值更多作用是在于设计,是设计者设计好后,让子类继承并实现抽象类()
4)抽象类,是考官比较爱问的知识点在框架和设计模式使用较多
三、抽象类的注意事项和细节讨论
1)抽象类不能被实例化
报错'A' is abstract; cannot be instantiated

2)抽象类不定要包含abstract方法。 也就是说抽象类可以没有abstract方法
同上 上图的抽象类A类并没有抽象方法
3)一旦类包含了abstract方法,则这个类必须声明为abstract

4) abstract只能修饰类和方法,不能修饰属性和其它的。

5)抽象类可以有任意成员(抽象类本质还是类),比如:非抽象方法、构造器、静态属性等等
注意不能实例对象。
public class AbstractDetail02 {
    public static void main(String[] args) {
        D.nihao();
    }
}
abstract class D{
    public int n1 = 10;
    public static String name = "yinhai";
    public D(){
    }
    {
    }
    static {
    }
    public abstract void hello();
    public void ok(){
    }
    public static void nihao (){
    }
}
 
6)抽象方法不能有主体,即不能实现如图所示abstract void aaa(){};
7)如果个类继承 了抽象类, 则它必须实现抽象类的所有抽象方法,除非它自己也声明为abstract类。
abstract class E{
    public abstract void hi();
}
abstract class F extends E{
    
}
class G extends E{
    @Override
    public void hi() {//这里G类子类实现了父类E的抽象方法,就是有方法体
        
    }
} 
8)抽象方法不能使用private,final和static来修饰,因为这些关键字都是和重写相违背的。
        
四、抽象类的课堂练习
1)题1.思考: abstract final class A{}能编译通过吗,why?
// 不能 因为抽象的本意是继承重写
//final修饰就已经是最终类了
2)题2,思考: abstract public static void test2(); 能编译通过吗,why?
//不能 静态的方法是不能重写的
3)题3,思考: abstract private void test30;能编译通过吗,why?
//不能 除了public以外都不行
4)编写一个Employee类,声明为抽象类,包含如下三个属性: name, id, salary.提供必要的构造器和抽象方法: work0. 对于Manager类来说,他既是员工,还具有奖金(bonus)的属性。请使用继承的思想,设计CommonEmployee类和Manager类.要求类中提供必要的方法进行属性访问,实现work().提示"经理/普通员工名字工作中....
    public double getSalary() {
        return salary;
    }
    public void setSalary(double salary) {
        this.salary = salary;
    }
}
class Maanager extends Employee{
    private double bonus;
    public Maanager(String name, String id, double salary, double bonus) {
        super(name, id, salary);
        this.bonus = bonus;
    }
    public void work(){
        System.out.println("经理 " + getName() + "正在工作中");
    }
}
class CommonEmployee extends Employee{
    public void work(){
        System.out.println("普通员工 " + getName() + "正在工作中");
    }
    public CommonEmployee(String name, String id, double salary) {
        super(name, id, salary);
    }
} 
 
五、模版设计模式
需求
1.有多个类,完成不同的任务job
2.要求统计各自得到各自完成任务的时间
3.编程实现
引出模版设计模式
public class TestTemplate {
    public static void main(String[] args) {
        AA aa = new AA();
        aa.job();
        BB bb = new BB();
        bb.job();
    }
}
class AA {
    public void job() {
        //得到开始的时间
        long start = System.currentTimeMillis();
        long num = 0;
        for (long i = 0; i < 1000000; i++) {
            num += i;
        }
        //得到结束的时间
        long end = System.currentTimeMillis();
        System.out.println("执行时间" + (end - start) + "ms");
    }
    public void job2(){
        //得到开始的时间
        long start = System.currentTimeMillis();
        long num = 0;
        for (long i = 0; i < 3000000; i++) {
            num += i;
        }
        //得到结束的时间
        long end = System.currentTimeMillis();
        System.out.println("执行时间" + (end - start) + "ms");
    }
}
class BB {
    public void job() {
        //得到开始的时间
        long start = System.currentTimeMillis();
        long num = 0;
        for (long i = 0; i < 2000000; i++) {
            num += i;
        }
        //得到结束的时间
        long end = System.currentTimeMillis();
        System.out.println("执行时间" + (end - start) + "ms");
    }
} 

如果在每个单类里面写job和计算时间的代码块,复用性会很差,所以就引出模版设计,设计一个父类,里面塞进去共有的计算时间的方法,然后塞入抽象job方法,因为每一个对象在运行的时候会从本类开始查找,所以构成了重写抽象job方法,所以这种设计理念就相当于把父类作为一个模版,其他继承。
所以改进该代码
public class TestTemplate {
    public static void main(String[] args) {
        AA aa = new AA();
        BB bb = new BB();
        aa.calWorkTime();
        bb.calWorkTime();
    }
}
abstract class Template{
    public abstract void job();
    public void calWorkTime(){
        long start = System.currentTimeMillis();
        job();//动态绑定机制,调用的是对象所在类内的方法,没有再往上查找
        long end = System.currentTimeMillis();
        System.out.println("执行时间" + (end - start) + "ms");
    }
}
class AA extends Template{
    public void job() {
        long num = 0;
        for (long i = 0; i < 1000000; i++) {
            num += i;
        }
    }
}
class BB extends Template{
    public void job() {
        long num = 0;
        for (long i = 0; i < 2000000; i++) {
            num += i;
        }
    }
} 
        



















