14.java中面向对象--抽象类与抽象方法(7)(附讲解与练习)

抽象类与抽象方法

abstract关键字的使用
1.abstract:抽象的
2.abstract可以用来修饰的结构:类、方法

  1. abstract修饰类:抽象类

此类不能实例化(创建不了对象,类中的方法仍可被子类调用
抽象类中一定有构造器,便于子类实例化时调用(涉及:子类对象实例化的全过程)
开发中,都会提供抽象类的子类,让子类对象实例化,完成相关的操作

  1. abstract修饰方法:抽象方法

抽象方法只有方法的声明,没有方法体
包含抽象方法的类,一定是一个抽象类。反之,抽象类中可以没有抽象方法的。
若子类重写了父类中的所有的抽象方法后,此子类方可实例化
若子类没有重写父类中的所有的抽象方法,则此子类也是一个抽象类,需要使用abstract修饰

abstract使用上的注意点:
 不能用abstract修饰变量、代码块、构造器;(构造器不能重写,只能重载)
 不能用abstract修饰私有方法、静态方法、 final的方法、 final的类

下面展示一些 AbstractTest

public class AbstractTest {
	public static void main(String[] args) {
		
		//一旦Person类抽象了,就不可实例化
//		Person p1 = new Person();
//		p1.eat();
		
	}
}

abstract class Creature{
	public abstract void breath();
}

abstract class Person extends Creature{
	String name;
	int age;
	
	public Person(){
		
	}
	public Person(String name,int age){
		this.name = name;
		this.age = age;
	}
	
	//不是抽象方法:
//	public void eat(){
//		
//	}
	//抽象方法
	public abstract void eat();
	
	public void walk(){
		System.out.println("人走路");
	}
	
	
}


class Student extends Person{
	
	public Student(String name,int age){
		super(name,age);
	}
	public Student(){
	}
	
	public void eat(){
		System.out.println("学生多吃有营养的食物");
	}

	@Override
	public void breath() {
		System.out.println("学生应该呼吸新鲜的没有雾霾的空气");
	}
}

练习:编写一个Employee类,声明为抽象类,包含如下三个属性: name, id,salary。
提供必要的构造器和抽象方法: work()。
对于Manager类来说,他既是员工,还具有奖金(bonus)的属性。
请使用继承的思想,设计CommonEmployee类和Manager类,要求类
中提供必要的方法进行属性访问。
下面展示一些 Employee

/*
 * 编写一个Employee类,声明为抽象类,
包含如下三个属性:name,id,salary。
提供必要的构造器和抽象方法:work()。
 * 
 * 
 */
public abstract class Employee {
	
	private String name;
	private int id;
	private double salary;
	public Employee() {
		super();
	}
	public Employee(String name, int id, double salary) {
		super();
		this.name = name;
		this.id = id;
		this.salary = salary;
	}
	
	public abstract void work();
}

下面展示一些 Manager extends Employee

/*
 * 对于Manager类来说,他既是员工,还具有奖金(bonus)的属性。
 */
public class Manager extends Employee{
	
	private double bonus;//奖金

	
	
	public Manager(double bonus) {
		super();
		this.bonus = bonus;
	}

	public Manager(String name, int id, double salary, double bonus) {
		super(name, id, salary);
		this.bonus = bonus;
	}

	@Override
	public void work() {
		System.out.println("管理员工,提供公司运行的效率");
	}
	
	
	
}

下面展示一些 CommonEmployee extends Employee

public class CommonEmployee extends Employee {

	@Override
	public void work() {
		System.out.println("员工在一线车间生产产品");
	}

}

下面展示一些 EmployeeTest

/*
 * 请使用继承的思想,设计CommonEmployee类和Manager类,要求类中提供必要的方法进行属性访问。
 */
public class EmployeeTest {
	public static void main(String[] args) {
		
		//多态
		Employee manager = new Manager("库克", 1001, 5000, 50000);
		
		manager.work();
		
		CommonEmployee commonEmployee = new CommonEmployee();
		commonEmployee.work();
		
	}
}

抽象类的匿名子类
下面展示一些 PersonTest

/*
 * 抽象类的匿名子类
 * 
 */
public class PersonTest {
	
	public static void main(String[] args) {
		
		method(new Student());//匿名对象
		
		Worker worker = new Worker();
		method1(worker);//非匿名的类非匿名的对象
		
		method1(new Worker());//非匿名的类匿名的对象
		
		System.out.println("********************");
		
		//创建了一匿名子类的对象:p,虽然new的是Person,但是创建的是一个子类,
		//将抽象类中的子类进行重写即可
		Person p = new Person(){

			@Override
			public void eat() {
				System.out.println("吃东西");
			}

			@Override
			public void breath() {
				System.out.println("好好呼吸");
			}
			
		};
		
		method1(p);
		
		System.out.println("********************");
		//创建匿名子类的匿名对象
		method1(new Person(){
			@Override
			public void eat() {
				System.out.println("吃好吃东西");
			}

			@Override
			public void breath() {
				System.out.println("好好呼吸新鲜空气");
			}
		});
	}
	
	
	public static void method1(Person p){
		p.eat();
		p.breath();
	}
	
	public static void method(Student s){
		
	}
}

class Worker extends Person{

	@Override
	public void eat() {
	}

	@Override
	public void breath() {
	}
	
}

多态的应用:模板方法设计模式(TemplateMethod)

抽象类体现的就是一种模板模式的设计,抽象类作为多个子类的通用模板,子类在抽象类的基础上进行扩展、改造,但子类总体上会保留抽象类的行为方式。

解决的问题:
 当功能内部一部分实现是确定的, 一部分实现是不确定的。这时可以把不确定的部分暴露出去,让子类去实现。
 换句话说,在软件开发中实现一个算法时,整体步骤很固定、通用,这些步骤已经在父类中写好了。但是某些部分易变,易变部分可以抽象出来,供不同子类实现。这就是一种模板模式。

下面展示一些 TemplateMethodTest

//抽象类的应用:模板方法的设计模式
public class TemplateMethodTest {

	public static void main(String[] args) {
	//多态
		BankTemplateMethod btm = new DrawMoney();
		btm.process();

		BankTemplateMethod btm2 = new ManageMoney();
		btm2.process();
	}
}
abstract class BankTemplateMethod {
	// 具体方法
	public void takeNumber() {
		System.out.println("取号排队");
	}

	public abstract void transact(); // 办理具体的业务 //钩子方法

	public void evaluate() {
		System.out.println("反馈评分");
	}

	// 模板方法,把基本操作组合到一起,子类一般不能重写
	public final void process() {
		this.takeNumber();

		this.transact();// 像个钩子,具体执行时,挂哪个子类,就执行哪个子类的实现代码

		this.evaluate();
	}
}

class DrawMoney extends BankTemplateMethod {
	public void transact() {
		System.out.println("我要取款!!!");
	}
}

class ManageMoney extends BankTemplateMethod {
	public void transact() {
		System.out.println("我要理财!我这里有2000万美元!!");
	}
}

下面展示一些 TemplateTest

/*
 * 抽象类的应用:模板方法的设计模式
 * 
 */
public class TemplateTest {
	public static void main(String[] args) {
		
		SubTemplate t = new SubTemplate();
		
		t.spendTime();
	}
}

abstract class Template{
	
	//计算某段代码执行所需要花费的时间
	public void spendTime(){
		
		long start = System.currentTimeMillis();
		
		this.code();//不确定的部分、易变的部分
		
		long end = System.currentTimeMillis();
		
		System.out.println("花费的时间为:" + (end - start));
		
	}
	
	public abstract void code();
	
	
}

class SubTemplate extends Template{

	@Override
	public void code() {
		
		for(int i = 2;i <= 1000;i++){
			boolean isFlag = true;
			for(int j = 2;j <= Math.sqrt(i);j++){
				
				if(i % j == 0){
					isFlag = false;
					break;
				}
			}
			if(isFlag){
				System.out.println(i);
			}
		}

	}
	
}

练习:编写工资系统, 实现不同类型员工(多态)的按月发放工资。 如果当月出现某个
Employee对象的生日, 则将该雇员的工资增加100元。
实验说明:
(1)定义一个Employee类,该类包含:private成员变量name,number,birthday,其中birthday 为MyDate类的对象;abstract方法earnings();toString()方法输出对象的name,number和birthday。
(2) MyDate类包含:private成员变量year,month,day ;toDateString()方法返回日期对应的字符串: xxxx年xx月xx日
(3)定义SalariedEmployee类继承Employee类,实现按月计算工资的员工处理。该类包括: private成员变量monthlySalary;实现父类的抽象方法earnings(),该方法返回monthlySalary值; toString()方法输出员工类型信息及员工的name, number,birthday。
( 4)参照SalariedEmployee类定义 类,实现按小时计算工资的
员工处理。该类包括:private成员变量wage和hour;实现父类的抽象方法earnings(),该方法返回wage*hour值;toString()方法输出员工类型信息及员工的name, number,birthday。
( 5)定义PayrollSystem类,创建Employee变量数组并初始化,该数组存放各类雇员对象的引用。利用循环结构遍历数组元素,输出各个对象的类型,name,number,birthday,以及该对象生日。当键盘输入本月月份值时,如果本月是某个Employee对象的生日,还要输出增加工资信息。
提示:
//定义People类型的数组People c1[]=new People[10];
//数组元素赋值
c1[0]=new People(“John”,“0001”,20);
c1[1]=new People(“Bob”,“0002”,19);
//若People有两个子类Student和Officer, 则数组元素赋值时, 可以使父类类型的数组元素指向子类。
c1[0]=new Student(“John”,“0001”,20,85.0);
c1[1]=new Officer(“Bob”,“0002”,19,90.5);

下面展示一些 Employee

package ToObject.Test5;

/**
 * @author dell
 * @version 11.0
 * @ClassnameEmployee
 * @Description Do
 * @date 2021/7/15 21:32
 * @Project Java2018
 **/

/*
定义一个Employee类,该类包含:private成员变量name,number,birthday,
其中birthday 为MyDate类的对象;abstract方法earnings();
toString()方法输出对象的name,number和birthday。
 */
public abstract class Employee {
    private String name;
    private int number;
    private MyDate birthday;

    public Employee(){
        super();
    }

    public Employee(String name, int number, MyDate birthday) {
        this.name = name;
        this.number = number;
        this.birthday = birthday;
    }

    public String getName() {
        return name;
    }

    public int getNumber() {
        return number;
    }

    public MyDate getBirthday() {
        return birthday;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setNumber(int number) {
        this.number = number;
    }

    public void setBirthday(MyDate birthday) {
        this.birthday = birthday;
    }

    public abstract double earnings();

    @Override
    public String toString() {
        return "Employee{" +
                "name='" + name + '\'' +
                ", number=" + number +
                ", birthday=" + birthday.toDateString() +
                '}';
    }
}


下面展示一些 内联代码片

// A code block
var foo = 'bar';
// An highlighted block
var foo = 'bar';

下面展示一些 MyDate

package ToObject.Test5;

/**
 * @author dell
 * @version 11.0
 * @Classname MyDate
 * @Description Do
 * @date 2021/7/15 21:33
 * @Project Java2018
 **/
/*
MyDate类包含:private成员变量year,month,day ;toDateString()方法返回日期对应的字符串: xxxx年xx月xx日
 */
public class MyDate {
    private int year;
    private int month;
    private int day;
    public MyDate(){
        super();
    }

    public MyDate(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }

    public int getYear() {
        return year;
    }

    public void setYear(int year) {
        this.year = year;
    }

    public int getMonth() {
        return month;
    }

    public void setMonth(int month) {
        this.month = month;
    }

    public int getDay() {
        return day;
    }

    public void setDay(int day) {
        this.day = day;
    }


    public String toDateString() {
        return year + "年" + month + "月" + day + "日";
    }
}

下面展示一些 SalariedEmployee extends Employee

package ToObject.Test5;

/**
 * @author dell
 * @version 11.0
 * @ClassnameSalariedEmployee
 * @Description Do
 * @date 2021/7/15 21:33
 * @Project Java2018
 **/
/*
定义SalariedEmployee类继承Employee类,实现按月计算工资的员工处理。
该类包括: private成员变量monthlySalary;实现父类的抽象方法earnings(),
该方法返回monthlySalary值; toString()方法输出员工类型信息及员工的name, number,birthday。
 */


public class SalariedEmployee extends Employee{
    private  double mothlySalary;
    public SalariedEmployee(){
        super();
    }

    public SalariedEmployee(String name, int number, MyDate birthday, double mothlySalary) {
        super(name, number, birthday);
        this.mothlySalary = mothlySalary;
    }

    public double getMothlySalary() {
        return mothlySalary;
    }



    public void setMothlySalary(double mothlySalary) {
        this.mothlySalary = mothlySalary;
    }


    public double earnings(){
        return mothlySalary;
    }
    public String toString(){
        return "SalariedEmployee[" + super.toString() + "]";
    }
}

下面展示一些 HourlyEmployee extends Employee

package ToObject.Test5;

/**
 * @author dell
 * @version 11.0
 * @ClassnameHourlyEmployee
 * @Description Do
 * @date 2021/7/15 21:34
 * @Project Java2018
 **/
/*
参照SalariedEmployee类定义 类,实现按小时计算工资的员工处理。
该类包括:private成员变量wage和hour;实现父类的抽象方法earnings(),该方法返回wage*hour值;
toString()方法输出员工类型信息及员工的name, number,birthday。
 */


public class HourlyEmployee extends Employee{
    private double wage;
    private int hour;

    public HourlyEmployee(){
        super();

    }

    public HourlyEmployee(String name, int number, MyDate birthday, double wage, int hour) {
        super(name, number, birthday);
        this.wage = wage;
        this.hour = hour;
    }

    public double earnings(){
        return (double) (wage * hour);


    }

    public String toString(){
        return "HourlyEmployee[" + super.toString() + "]";
    }
}

下面展示一些 PayrollSystem

package ToObject.Test5;

/**
 * @author dell
 * @version 11.0
 * @ClassnamePayrollSystem
 * @Description Do
 * @date 2021/7/15 21:35
 * @Project Java2018
 **/

/*
定义PayrollSystem类,创建Employee变量数组并初始化,
该数组存放各类雇员对象的引用。利用循环结构遍历数组元素,
输出各个对象的类型,name,number,birthday,以及该对象生日。
当键盘输入本月月份值时,如果本月是某个Employee对象的生日,还要输出增加工资信息。
 */

import java.util.Calendar;

public class PayrollSystem {
    public static void main(String[] args) {
//        //方式一,采取键盘输入的方式
//
//        Scanner scanner = new Scanner(System.in);
//        System.out.println("请输入员工身份月:");
//        int month = scanner.nextInt();

        //方式二:调用日历功能
        Calendar calendar = Calendar.getInstance();
        int month = calendar.get(Calendar.MONTH);//获取当前的月份
//		System.out.println(month);//一月份:0



        Employee [] emps = new Employee[2];
        emps[0] = new SalariedEmployee("均可",1001,new MyDate(1992,7,12),50000);
        emps[1] = new HourlyEmployee("死神",1002,new MyDate(1994,5,2),129,240);

        for (int i = 0;i < emps.length;i++){
            System.out.println(emps[i]);
            double salary = emps[i].earnings();
            System.out.println("月工资为:" + salary);


方式一,采取键盘输入的方式
//            if (month == emps[i].getBirthday().getMonth()){
//                System.out.println("生日快乐,奖励一百元!当月工资为:" + (emps[i].earnings() + 100));
//            }

            //方式二:调用日历功能
            if ((month + 1) == emps[i].getBirthday().getMonth()){
                System.out.println("生日快乐,奖励一百元!当月工资为:" + (emps[i].earnings() + 100));
            }
        }
    }

}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

原来如此呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值