抽象类与抽象方法

abstract : 抽象的

abstract可以用来修饰:类、方法

abstract修饰类

> 此类称为抽象类

> 抽象类不能实例化

> 抽象类中是包含构造器的,因为子类对象实例化时,需要直接或间接的调用到父类的构造器

> 抽象类中可以没有抽象方法。反之,抽象方法所在的类,一定是抽象类。

抽象方法

> 此方法即为抽象方法

> 抽象方法只有方法的声明,没有方法体

> 抽象方法其功能是确定的(通过方法的声明即可确定),只是不知道如何具体实现(体现为:没有方法体)

> 子类必须重写父类中的所有抽象方法之后,方可实例化,否则此子类仍然是一个抽象类(还有抽象方法)

public abstract class Person { //抽象类
String name;
int age; public Person(){
} public Person(String name, int age){
this.name = name;
this.age = age;
}
public abstract void eat(); //抽象方法 public void sleep(){
System.out.println("人会睡觉");
}
}
public class Student extends Person{
String school; public Student() {
} public Student(String name, int age, String school) {
super(name, age);
this.school = school;
} public void eat(){
System.out.println("学生多吃有营养的食物");
} public void sleep() {
System.out.println("学生要保证睡眠");
}
}
public class AbstractTest {
public static void main(String[] args) {
// Person p1 = new Person(); Person设置为抽象类后不可以再创建对象 Student s1 = new Student();
s1.sleep();
}
}

 abstract不能使用的场景

abstract不能修饰:  属性、构造器、代码块等

abstract不能与哪些关键字共用:(自洽)

不能用abstract修饰私有方法、静态方法、final的方法、final的类。

> 私有方法不能重写

> 避免静态方法使用此类进行调用

> final的方法不能够被重写

> final修饰的类不能有子类

模板方法设计模式(TemplateMethod)

解决的问题:

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

在此我们设计了一个模板方法,用于计算代码运行的时间,这里的code可以在后续子类中重写来计算code运行所需要花费的时间。

/**
* 抽象应用案例: 模板方法的设计模式
*/
public class TemplateTest {
public static void main(String[] args) {
PrintPrimeNumber p = new PrintPrimeNumber();
p.spendTime();
}
}
abstract class Template{
//计算某段代码执行需要花费的时间
public void spendTime(){
long start = System.currentTimeMillis(); code(); long end = System.currentTimeMillis(); System.out.println("花费的时间为:" + (end -start)); } public abstract void code();
} class PrintPrimeNumber extends Template{
public void code(){
for (int i = 2; i <= 100000; 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。

public abstract class Employee {
private String name;
private int number;
private MyDate birthday; public Employee() {
} public Employee(String name, int number, MyDate birthday) {
this.name = name;
this.number = number;
this.birthday = birthday;
} public abstract double earnings(); //抽象方法 返回工资数额 @Override
public String toString() {
return "name='" + name + "', number=" + number +
", birthday=" + birthday ;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public int getNumber() {
return number;
} public void setNumber(int number) {
this.number = number;
} public MyDate getBirthday() {
return birthday;
} public void setBirthday(MyDate birthday) {
this.birthday = birthday;
}
}

(2) MyDate类包含:
private成员变量year,month,day;
提供必要的构造器;
toDateString()方法返回日期对应的字符串,xxxx年xx月xx日

public class MyDate {
private int year;
private int month;
private int day; public MyDate() {
} public MyDate(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
} @Override
public String toString() {
return year + "年" + month + "月" + 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;
}
}

3)定义SalariedEmployee类继承Employee类,实现按月计算工资的员工处理
该类包括: private成员变量monthlySalary;
提供必要的构造器;
实现父类的抽象方法earnings()该方法返回monthlySalary值;
toString()方法输出员工类型信息及员工的name,number,birthday。比如: SalariedEmployee[name = ' ' ,number = , birthday = xxxx年xx月xx日]

public class SalariedEmployee extends Employee{
private double monthlySalary; public SalariedEmployee() {
} public SalariedEmployee(String name, int number, MyDate birthday, double monthlySalary) {
super(name, number, birthday);
this.monthlySalary = monthlySalary;
} @Override
public double earnings() {
return monthlySalary;
} @Override
public String toString() {
return "SalariedEmployee[ " + super.toString() + "]";
} public double getMonthlySalary() {
return monthlySalary;
} public void setMonthlySalary(double monthlySalary) {
this.monthlySalary = monthlySalary;
}
}

(4)参照SalariedEmployee类定义HourlyEmployee类,实现按小时计算工资的员工处理。该类包括
private成员变量wage和hour;
提供必要的构造器;
实现父类的抽象方法earnings()该方法返回waqe*hour值;
toString0方法输出员工类型信息及员工的name, number, birthday。

public class HourlyEmployee extends Employee{
private double wage;
private int hour; public HourlyEmployee() {
} public HourlyEmployee(String name, int number, MyDate birthday, double wage, int hour) {
super(name, number, birthday);
this.wage = wage;
this.hour = hour;
} @Override
public double earnings() {
return wage * hour;
} @Override
public String toString() {
return "HourlyEmployee = [" + super.toString() + "]";
} public double getWage() {
return wage;
} public void setWage(double wage) {
this.wage = wage;
} public int getHour() {
return hour;
} public void setHour(int hour) {
this.hour = hour;
}
}

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

import java.util.Scanner;
public class PayrollSystem {
public static void main(String[] args) {
Scanner scan = new Scanner(System.in);
Employee[] employees = new Employee[2];
employees[0] = new SalariedEmployee("小明",10001,new MyDate(2000,12,25),3000);
employees[1] = new HourlyEmployee("小红",10002,new MyDate(2001,1,9),100,32);
System.out.println("请输入当前的月份:");
int month = scan.nextInt();
for (int i = 0; i < employees.length; i++){
System.out.println(employees[i].toString());
if (month == employees[i].getBirthday().getMonth()){
System.out.println("此月生日!!");
System.out.println("工资:" + (employees[i].earnings() + 100));
}else{
System.out.println("工资:" + employees[i].earnings());
}
}
}
}

 接口

接口的理解: 接口的本质是 契约、标准、规范。就像法律一样,制定好以后大家都要遵守

定义接口的关键字:interface

接口内部结构的说明:

> 可以声明:

  属性: 必须使用public static final修饰

  方法:jdk8之前:声明抽象方法,修饰为public abstract

     jdk8:声明静态方法、默认方法

       jdk9:声明私有方法

> 不可以声明: 构造器、代码块等

接口与类的关系 :  实现关系

格式: class A extends SuperA implements B,C{ }

A相较于SuperA来讲,叫做子类。

A相较于B,C来讲,叫做实现类。

满足此关系后,说明:

> 类可以实现多个接口。

> 类针对于接口的多实现,一定程度上就弥补了类的单继承的局限性。

> 类必须将实现的接口中的所有的抽象方法都重写(或实现),方可实例化。否则,此实现类必须声明为抽象类。

Java基础12的更多相关文章

  1. Java基础12:深入理解Class类和Object类

    更多内容请关注微信公众号[Java技术江湖] 这是一位阿里 Java 工程师的技术小站,作者黄小斜,专注 Java 相关技术:SSM.SpringBoot.MySQL.分布式.中间件.集群.Linux ...

  2. 【java基础 12】HashMap中是如何形成环形链表的?

    导读:经过前面的博客总结,可以知道的是,HashMap是有一个一维数组和一个链表组成,从而得知,在解决冲突问题时,hashmap选择的是链地址法.为什么HashMap会用一个数组这链表组成,当时给出的 ...

  3. Java基础12 类型转换与多态

    链接地址:http://www.cnblogs.com/vamei/archive/2013/04/01/2992662.html 作者:Vamei 出处:http://www.cnblogs.com ...

  4. java基础12 抽象类(及关键字:abstract)

    抽象类:abstract 1.应用的场景 我们描述一类事物时,存在着某种行为,但这种行为目前不具体,那么我们就可以抽取这种行为的声明,但是不去实现这种行为,我们就需要使用抽象类. 2.抽象的好处 强制 ...

  5. Java基础12 类型转换与多态(转载)

    类型检查 Java的任意变量和引用经过类型声明(type declaration),才能使用.我们之前见过对象数据.类数据.方法参数.方法返回值以及方法内部的自动变量,它们都需要声明其类型.Java是 ...

  6. Java基础12一IO流

    1.IO流的原理 利用数据通道实现程序和数据源之间数据的的读写操作.  2.IO流分类 输入流.输出流.字节流.字符流.节点流.过滤流  3.InputStream 字节输入流 实现类FileInpu ...

  7. java基础(12):构造方法、this、super

    1. 构造方法 我们对封装已经有了基本的了解,接下来我们来看一个新的问题,依然以Person为例,由于Person中的属性都被private了,外界无法直接访问属性,必须对外提供相应的set和get方 ...

  8. android基础篇------------java基础(12)(多线程操作)

    <一>基本概念理解 1.什么是进程? 进程就是在某种程度上相互隔离,独立运行的程序.一般来说,系统都是支持多进程操作的,这所谓的多进程就是让系统好像同时运行多个程序. 2.什么是线程呢? ...

  9. 2015年12月28日 Java基础系列(六)流

    2015年12月28日 Java基础系列(六)流2015年12月28日 Java基础系列(六)流2015年12月28日 Java基础系列(六)流

  10. 050 01 Android 零基础入门 01 Java基础语法 05 Java流程控制之循环结构 12 continue语句

    050 01 Android 零基础入门 01 Java基础语法 05 Java流程控制之循环结构 12 continue语句 本文知识点:continue语句 continue语句 continue ...

随机推荐

  1. ComfyUI基础篇:为什么要学 ComfyUI?

    前言: 在AI生成图像领域,有许多产品,例如 Midjourney 和 Stability AI 等.为什么要学习 ComfyUI 呢?我斗胆带大家一起分析一下. 目录 1.Midjourney VS ...

  2. 你真的了解Java内存模型JMM吗?

    哈喽,大家好,我是世杰. 本文我为大家介绍面试官经常考察的「Java内存模型JMM相关内容」 面试连环call 什么是Java内存模型(JMM)? 为什么需要JMM? Java线程的工作内存和主内存各 ...

  3. [oeasy]python0144_try的完全体_否则_else_最终_finally

    try的完全体 回忆上次内容   上次细化了主控程序(main.py) 导入(get_fruits.py) 处理(process.py) 输出(output.py)   使用了 try 结构 try ...

  4. oeasy教您玩转vim - 40 - # 复制粘贴

    ​ 复制粘贴 回忆上节课内容 我们上次的内容是粘贴 小写p意味着在光标下面或者后面粘贴 大写P意味着在光标上面或者前面粘贴 p的意思是放上去,就是put 把什么放上去呢? 把 reg 中 " ...

  5. Windows11下使用VcXsrv+xfce4实现图形化窗口

    通过 sudo apt-get install xfce4-terminal sudo apt-get install xfce4 sudo service dbus restart 来安装所需要的软 ...

  6. Prometheus 使用Python推送指标数据到Pushgateway

    使用Python推送指标数据到Pushgateway 需求描述 实践环境 Python 3.6.5 Django 3.0.6 prometheus-client 0.11.0 代码实现 !/usr/b ...

  7. 垃圾回收器比较:CMS 和 G1

    前言 在查看系统内存监控的过程中,发现有几台机器的内存使用率一直很高,而且是呈现一个不太正常的高度,初始以为是 GC 不完全,也就是 JVM 内有大量对象不能回收,于是采用 Arthas 诊断查看一下 ...

  8. Linux podman容器实验

    要求 1.登录到指定的镜像仓库拉取镜像 2.以普通用户contsvc运行 3.配置 systemd-journald 服务,要求永久保留日志数据(物理机,创建文件mkdir /var/log/jour ...

  9. hive导入mysql

    hive测试--HIVE数据分析02 题目: 4.处理结果入库:(在虚拟机安装mysql)    将上述统计分析的结果数据保存到mySQL数据库中. #text3_1入库 #1.添加驱动,在hive的 ...

  10. Postman汉化成中文版

    postman安装默认是英文版,为使用方便使用汉化包转成中文版 1.查看本地安装的postman版本:Settings->About 2.根据postman的版本下载对应的汉化包,汉化包网址 3 ...