abstract:

  1. * abstract修饰类: 抽象类
  2. * > 此类不能被实例化
  3. * > 抽象类中一定要有构造器, 便于子类对象实例时调用(涉及子类对象实例化过程)
  4. * > 开发中,都会提供抽象类的子类,让子类对象实例化.完成相关操作
  5. *
  6. *
  7. *
  8. * abstract修饰方法: 抽象方法
  9. * > 只有方法的声明没有方法体
  10. * > 如果类中有抽象方法 那么这个类必须也是抽象的(包含抽象方法的类一定是抽象类)
  11. * > 若子类重写了父类中的所有的抽象方法后,方可实例化, 若子类没有重写父类中的抽象方法 则子类也是个抽象类 也需要abstract修饰
  1. * abstract 不能用来修饰: 属性, 构造器等结构
    * abstract 不能修饰: 私有方法 静态方法(加上static不认为是重写), final方法, final
  1. abstract class PP{
  2. int age;
  3.  
  4. public PP(){};
  5.  
  6. public PP(int age){
  7. this.age = age;
  8. }
  9.  
  10. public abstract void eat(); // 父类有抽象方法 子类必须重写这个方法
  11. }
  12.  
  13. class Pson extends PP{
  14.  
  15. public void eat(){ // 重写父类的抽象方法
  16.  
  17. }
  18.  
  19. }

由于abstract 抽象类不可以被实例化 那么我们还有一种方式是正常的这种看着像是实例化抽象类 其实不是实例化抽象类 而是建立了一个匿名对象,在匿名对象中重写了抽象类中的抽象方法

  1. public class abstractTest {
  2. public static void main(String[] args) {
  3. // 创建一个匿名对象empty
  4. Empty empty = new Empty() { //因为Empty是抽象类abstract 的 不可以被实例化 所以我们现在造的是匿名的Empty 对象
  5. @Override
  6. public void work() {
  7. System.out.println("我是匿名对象的work方法");
  8. }
  9. };
  10. method1(empty); // 我是匿名对象的work方法
  11. }
  12.  
  13. public static void method1(Empty empty){
  14. empty.work();
  15. }
  16. }
  17.  
  18. abstract class Empty{
  19. private String name;
  20. private int id;
  21. private double salary;
  22. public Empty(){};
  23. public Empty(String name, int id, double salary){
  24. super();
  25. this.name = name;
  26. this.id = id;
  27. this.salary = salary;
  28. }
  29.  
  30. public abstract void work(); // 抽象方法必须要是无方法体的
  31. }
  32.  
  33. class Mang extends Empty{
  34.  
  35. private double bonus;
  36. public void work(){
  37. System.out.println("可以跑");
  38. }
  39. }
  40.  
  41. class Wolk extends Empty{
  42. public void testOne(){
  43. System.out.println("test");
  44. }
  45.  
  46. public void work(){
  47. System.out.println("wolk的work");
  48. }
  49. }

为什么abstract不可以修饰属性

  1. abstract修饰的内容都是暂未被实现的,比如类、方法。
  2. 属性之所以不能被abstract修饰,是因为属性不存在"尚未被实现"的状态。
  3. 比如你可能会联想到int age; 或是String name; 但可惜的是,在申明变量时,int会默认给age赋予初始值0
    String会默认给name赋予初始值""。因此属性达不到"尚未被实现的状态",从而不能被abstract修饰

为什么不可以修饰私有方法和构造器

  1. 因为抽象类被继承后 子类要重写抽象类中的所有的抽象方法,但是私有的方法在权限上是不可以被调用重写的 所有不可以修饰私有方法不然子类无法重写此方法,构造器时子类要去调用的 而不是要重写完才去调用的 所以构造器也不可以被abstract

抽象类的使用场景

  1. 抽象类体现的就是一种模板模式的设计,抽象类作为多个子类的通用模 板,子类在抽象类的基础上进行扩展、改造,但子类总体上会保留抽象 类的行为方式
  2.  
  3. 当功能内部一部分实现是确定的,一部分实现是不确定的。这时可以 把不确定的部分暴露出去,让子类去实现。
  4. 换句话说,在软件开发中实现一个算法时,整体步骤很固定、通用, 这些步骤已经在父类中写好了。但是某些部分易变,易变部分可以抽 象出来,供不同子类实现。这就是一种模板模式

思考题:

1: 为什么抽象类不可以被final声明:

  1. 因为抽象类是需要被继承的 如果被final声明就不可以被继承此类就是个无用类了

2: 一个抽象类可以定义构造器吗?

  1. 可以, 因为抽象类虽然不可以实例化 ,但是子类实例化的时候需要调用父类的构造器

3:

  1. 编写一个Employee类,声明为抽象类, 包含如下三个属性:nameidsalary 提供必要的构造器和抽象方法:work()。 对于Manager类来说,他既是员工,还具有奖金(bonus)的属性。
  2. 请使用继承的思想,设计CommonEmployee类和Manager类,要求类 中提供必要的方法进行属性访问。
  1. abstract class Empty{
  2. private String name;
  3. private int id;
  4. private double salary;
  5. public Empty(){};
  6. public Empty(String name, int id, double salary){
  7. super();
  8. this.name = name;
  9. this.id = id;
  10. this.salary = salary;
  11. }
  12.  
  13. public abstract void work(); // 抽象方法必须要是无方法体的
  14. }
  15.  
  16. class Mang extends Empty{
  17.  
  18. private double bonus;
  19. public void work(){
  20. System.out.println("可以跑");
  21. }
  22. }

结果

作业:

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

解答:

  1. import java.util.Date;
  2.  
  3. public abstract class Emplpyee {
  4. private String name; // 名字
  5. private double number; // 工作时间
  6. private MyDate birthday; //生日
  7.  
  8. abstract double earnings();
  9.  
  10. public String getName() {
  11. return name;
  12. }
  13.  
  14. public void setName(String name) {
  15. this.name = name;
  16. }
  17.  
  18. public double getNumber() {
  19. return number;
  20. }
  21.  
  22. public void setNumber(double number) {
  23. this.number = number;
  24. }
  25.  
  26. public MyDate getBirthday() {
  27. return birthday;
  28. }
  29.  
  30. public void setBirthday(MyDate birthday) {
  31. this.birthday = birthday;
  32. }
  33.  
  34. public Emplpyee(){}
  35. public Emplpyee(String name, int number, MyDate birthday){
  36. this.name = name;
  37. this.number = number;
  38. this.birthday = birthday;
  39. }
  40.  
  41. @Override
  42. public String toString() {
  43. return
  44. "name='" + name + '\'' +
  45. ", number=" + number +
  46. ", birthday=" + birthday.toDateString() ; // 要去调用我们自定义的 DateString 不然就是显示的内存地址值了
  47. }
  48.  
  49. }
  50.  
  51. class MyDate {
  52. private int year;
  53. private int month;
  54. private int day;
  55.  
  56. public int getYear() {
  57. return year;
  58. }
  59.  
  60. public void setYear(int year) {
  61. this.year = year;
  62. }
  63.  
  64. public int getMonth() {
  65. return month;
  66. }
  67.  
  68. public void setMonth(int month) {
  69. this.month = month;
  70. }
  71.  
  72. public int getDay() {
  73. return day;
  74. }
  75.  
  76. public void setDay(int day) {
  77. this.day = day;
  78. }
  79.  
  80. public MyDate(){};
  81. public MyDate(int year ,int month, int day){
  82. this.year = year;
  83. this.month = month;
  84. this.day = day;
  85. }
  86.  
  87. public String toDateString(){
  88. return year + "年" + month + "月" + day + "日";
  89. }
  90. }

Employee

  1. public class SalariedEmployee extends Emplpyee {
  2. private double monthlySalary; // 月工资
  3.  
  4. public SalariedEmployee(String name, int number, MyDate birthday) {
  5. super(name, number, birthday);
  6. }
  7.  
  8. public SalariedEmployee(String name, int number, MyDate birthday, double monthlySalary) {
  9. super(name, number, birthday);
  10. this.monthlySalary = monthlySalary;
  11. }
  12.  
  13. @Override
  14. public double earnings() {
  15. return monthlySalary;
  16. }
  17.  
  18. @Override
  19. public String toString() {
  20. return "SalariedEmployee[" + super.toString() + "]";
  21. }
  22.  
  23. public double getMonthlySalary() {
  24. return monthlySalary;
  25. }
  26.  
  27. public void setMonthlySalary(double monthlySalary) {
  28. this.monthlySalary = monthlySalary;
  29. }
  30. }

SalariedEmployee

  1. public class HourlyEmployee extends Emplpyee {
  2. private int wange; // 月工资
  3. private int hour; //月小时数目
  4.  
  5. public HourlyEmployee(String name, int number, MyDate birthday) {
  6. super(name, number, birthday);
  7. }
  8.  
  9. public HourlyEmployee(String name, int number, MyDate birthday, int wange, int hour) {
  10. super(name, number, birthday);
  11. this.wange = wange;
  12. this.hour = hour;
  13. }
  14.  
  15. public double earnings() {
  16. return wange * hour;
  17. }
  18.  
  19. @Override
  20. public String toString() {
  21. return "HourlyEmployee[" + super.toString() + "]";
  22. }
  23. }

HourlyEmployee

  1. import java.util.Scanner;
  2.  
  3. public class PayrollSystem {
  4. public static void main(String[] args) {
  5. Scanner scanner = new Scanner(System.in);
  6. int month = scanner.nextInt();
  7. Emplpyee emplpyee [] = new Emplpyee[2];
  8. emplpyee[0] = new SalariedEmployee("老王", 30, new MyDate(1995, 9, 30), 12000);
  9. emplpyee[1] = new HourlyEmployee("隔壁老王", 160, new MyDate(1992, 5, 25),50, 260 );
  10.  
  11. for(int i = 0; i < emplpyee.length; i++){
  12. System.out.println(emplpyee[i]);
  13. double salary = emplpyee[i].earnings();
  14. System.out.println("月薪资为:"+salary);
  15.  
  16. if(month == emplpyee[i].getBirthday().getMonth()){
  17. System.out.println("生日快乐 奖励100");
  18. }
  19. }
  20. }
  21. }

PayrollSystem

Practise

  1. abstract 能修饰哪些结构? 修饰以后,有什么特点?
  1. abstract 可以修饰类和方法
  2.  
  3. 修饰后 类不可以实例化 需要提供子类来实例化 这个抽象类要有构造器来便于子类的实例化
  4.  
  5. 抽象方法 只有方法的声明没有方法体具体的执行,需要子类去实现
  1. 接口是否能继承接口? 抽象类是否能实现(implements)接口? 抽象类是否能继承非抽象的类?

    1. 接口的继承用extends来继承 抽象类能实现接口,抽象来能继承非抽象类
    1. . 声明抽象类,并包含抽象方法。测试类中创建一个继承抽象类的匿名子类的对象
    2. abstract AA{
    3. public abstract void m();
    4. }
    5. main(){
    6. AA a = new AA(){
    7. public void m(){
    8.  
    9. }
    10. };
    11. a.m();
    12.  
    13. }
    14.  
    15. class Person{
    16. String name;
    17. public void eat(){}
    18. }
    19.  
    20. main(){
    21. Person p = new Person(){
    22. public void eat(){}
    23. };
    24. p.eat();
    25. }
    26.  
    27. . 抽象类和接口有哪些共同点和区别?
    28.  
    29. 相同点:不能实例化,都可以被继承
    30. 不同点:抽象类:有构造器。 接口:不能声明构造器
    31. 多继承vs 单继承 抽象类是单继承接口是多继承
    32.  
    33. . 如何创建静态成员内部类和非静态成员内部类的对象?
    34. Person static Dog Bird
    35. Person.Dog dog = new Person.Dog();
    36. Person p = new Person();
    37. Person.Bird bird = p.new Bird();

抽象类 抽象方法 abstract的更多相关文章

  1. Java 面向对象编程 抽象类 抽象方法 abstract

    抽象类 通过abstract定义的方法是抽象方法,它只有定义,没有实现.抽象方法定义了子类必须实现的接口规范: 定义了抽象方法的class必须被定义为抽象类,从抽象类继承的子类必须实现抽象方法: 如果 ...

  2. Java abstract关键字 抽象类 抽象方法

    用 abstract 修饰的类是抽象类,它不能生成对象 含有抽象方法的类称为抽象类 抽象方法使用 abstract 修饰 抽象类不能直接实例化,只能由子类实例化 举例 abstract class T ...

  3. 面试题之——抽象类(abstract class)与接口(interface)的区别

    抽象类可以有构造方法,接口中不能有构造方法.(虽然抽象类有构造方法,但它也不能被实例化) 抽象类中可以有普通成员变量,接口中没有普通成员变量. 抽象类和接口中都可以包含静态成员变量.抽象类中的静态成员 ...

  4. 抽象类(abstract class)和 接口(interface)

    PHP中的 抽象类(abstract class)和 接口(interface) 一. 抽象类abstract class 1 .抽象类是指在 class 前加了 abstract 关键字且存在抽象方 ...

  5. 面向对象_06【抽象类:abstract、接口:interface、实现:implements】

    抽象类:abstract抽象:没有足够的描述功能,事物不具体的描述,却又有共性. 特点: 1,方法只有声明没有实现时,该方法就是抽象方法,需要被abstract修饰,抽象方法必须定义在抽象类中,该类也 ...

  6. 虚方法(virtual)和抽象方法(abstract)的和接口(interface)的区别

    虚方法(virtual)和抽象方法(abstract)的区别 2017年06月15日 13:41:26 阅读数:65 注:本文转载自 http://www.cnblogs.com/michaelxu/ ...

  7. 抽象类(abstract class)

    package com.bjsxt.oop.abstractClass; //抽象类 public abstract class Animal { //因为父类的方法总是被重写 那就没写的必要了 但是 ...

  8. c# 抽象类(abstract)

    using System; using System.Collections.Generic; using System.Linq; using System.Text; //抽象类(abstract ...

  9. 抽象类(abstract)【转】

    抽象类(abstract) abstract修饰符可以和类.方法.属性.索引器及事件一起使用.在类声明中使用abstract修饰符以指示某个类只能是其它类的基类.标记为抽象或包含在抽象类中的成员必须通 ...

随机推荐

  1. < python PIL - 批量图像处理 - 生成自定义大小图像 >

    < python PIL - 批量图像处理 - 生成自定义大小图像 > 直接用python自带的PIL图像库,对一个文件夹下所有jpg/png的图像进行自定义像素变换 from PIL i ...

  2. 机器学习 coursera_ML

    在开始看之前,浏览器一直出现缓冲问题,是配置文件设置的不对,最后搞定,高兴!解决方法如下: 1.到C:\Windows\System32\drivers\etc下找到host文件,并以文本方式打开, ...

  3. [功能集锦] 003 - 一键生成mysql数据字典/数据库速查表

    写在前面: 因为工作时候经常遇到半路接手项目的情况,由于年代久远,数据库字典这块经常缺失.故写此篇,以便复用,也希望对大家有点帮助. 随笔内容不高级,如有不妥,不吝指正. ps:有另一篇详细随笔可以参 ...

  4. LeetCode #657. Robot Return to Origin 机器人能否返回原点

    https://leetcode-cn.com/problems/robot-return-to-origin/ 设置 flagUD 记录机器人相对于原点在纵向上的最终位置 flagRL 记录机器人相 ...

  5. 点分治题单(来自XZY)

    点分治题单(来自XZY) 静态点分治 [x] 洛谷 P3806 [模板]点分治1 [x] 洛谷 P4178 Tree [x] 洛谷 P2634 [国家集训队]聪聪可可 [x] 洛谷 P4149 [IO ...

  6. Java中POJO及其细分XO、DAO的概念

    各层命名规约: A) Service/DAO 层方法命名规约 1) 获取单个对象的方法用 get 做前缀. 2) 获取多个对象的方法用 list 做前缀. 3) 获取统计值的方法用 count 做前缀 ...

  7. SpringCloud Caused by: org.springframework.boot.web.server.WebServerException: Unable to start embedded

    出现此问题,有可能是spring cloud 与spring boot 版本不匹配引发的问题,此次用的版本是:Finchley.RC1 经过一番关键字查找,发现spring cloud 与spring ...

  8. 源码分析--ConcurrentHashMap与HashTable(JDK1.8)

    ConcurrentHashMap和Hashtable都是线程安全的K-V型容器.本篇从源码入手,简要说明它们两者的实现原理和区别. 与HashMap类似,ConcurrentHashMap底层也是以 ...

  9. php中__call与__callstatic()使用

    php 5.3 后新增了 __call 与__callStatic 魔法方法. __call 当要调用的方法不存在或权限不足时,会自动调用__call 方法. __callStatic 当调用的静态方 ...

  10. SpringMVC表单或Json中日期字符串与JavaBean的Date类型的转换

    SpringMVC表单或Json中日期字符串与JavaBean的Date类型的转换 场景一:表单中的日期字符串和JavaBean的Date类型的转换 在使用SpringMVC的时候,经常会遇到表单中的 ...