首页 > 编程语言 >Java 面向对象编程进阶四(多态、抽象方法抽象类)

Java 面向对象编程进阶四(多态、抽象方法抽象类)

时间:2024-03-21 11:02:43浏览次数:27  
标签:Java String void 多态 面向对象编程 抽象类 方法 public

目录

多态(polymorphism)

多态和类型转换

对象的转型(casting)

 类型转换异常

向下转型中使用 instanceof

 final 关键字

抽象方法和抽象类

抽象类和抽象方法的基本用法


多态(polymorphism)

        多态指的是同一个方法调用,由于对象不同可能会有不同的行为。现实生活中,同一个 方法,具体实现会完全不同。 比如:同样是调用人的“休息”方法,张三是睡觉,李四是旅 游,高淇老师是敲代码,数学教授是做数学题; 同样是调用人“吃饭”的方法,中国人用 筷子吃饭,英国人用刀叉吃饭,印度人用手吃饭。

        多态的要点:

  1. 多态是方法的多态,不是属性的多态(多态与属性无关)。
  2. . 多态的存在要有 3 个必要条件:继承,方法重写,父类引用指向子类对象。
  3. 父类引用指向子类对象后,用该父类引用调用子类重写的方法,此时多态就出现了。

多态和类型转换

public class TestPolym {
    public static void main(String[ ] args) {
        Animal a1 = new Cat(); // 向上可以自动转型
        //传的具体是哪一个类就调用哪一个类的方法。大大ᨀ高了程序的可扩展性。
        animalCry(a1);
        Animal a2 = new Dog();
        animalCry(a2);//a2 为编译类型,Dog 对象才是运行时类型。
        /*编写程序时,如果想调用运行时类型的方法,只能进行强制类型转换。
         * 否则通不过编译器的检查。*/
        Dog dog = (Dog)a2;//向下需要强制类型转换
        dog.seeDoor();
    }
    // 有了多态,只需要让增加的这个类继承 Animal 类就可以了。
    static void animalCry(Animal a) {
        a.shout();
    }
    /* 如果没有多态,我们这里需要写很多重载的方法。
    * 每增加一种动物,就需要重载一种动物的喊叫方法。非常麻烦。
    static void animalCry(Dog d) {
        d.shout();
    }
     static void animalCry(Cat c) {
        c.shout();
    }*/
}
class Animal {
    public void shout() {
        System.out.println("叫了一声!");
    }
}
class Dog extends Animal {
    public void shout() {
        System.out.println("旺旺旺!");
    }
    public void seeDoor() {
        System.out.println("看门中....");
    }
}
class Cat extends Animal {
    public void shout() {
        System.out.println("喵喵喵喵!");
    }
}

        这是多态最为多见的一种用法,即父类引用做方法的形参,实参可 以是任意的子类对象,可以通过不同的子类对象实现不同的行为方式。

        由此,我们可以看出多态的主要优势是ᨀ高了代码的可扩展性,符合开闭原则。但是多 态也有弊端,就是无法调用子类特有的功能,比如,我不能使用父类的引用变量调用 Dog 类特有的 seeDoor()方法。

对象的转型(casting)

        父类引用指向子类对象,我们称这个过程为向上转型,属于自动类型转换。

        向上转型后的父类引用变量只能调用它编译类型的方法,不能调用它运行时类型的方 法。这时,我们就需要进行类型的强制转换,我们称之为向下转型!

public class TestCasting {
    public static void main(String[ ] args) {
        Object obj = new String("北京尚学堂"); // 向上可以自动转型
        // obj.charAt(0) 无法调用。编译器认为 obj 是 Object 类型而不是 String 类型
        /* 编写程序时,如果想调用运行时类型的方法,只能进行强制类型转换。
         * 不然通不过编译器的检查。 */
        String str = (String) obj; // 向下转型
        System.out.println(str.charAt(0)); // 位于 0 索引位置的字符
        System.out.println(obj == str); // true.他们俩运行时是同一个对象
    }
}

在向下转型过程中,必须将引用变量转成真实的子类类型(运行时类型)否则会出现类 型转换异常 ClassCastException。       

 类型转换异常

public class TestCasting2 {
    public static void main(String[ ] args) {
        Object obj = new String("北京尚学堂");
        //真实的子类类型是 String,但是此处向下转型为 StringBuffer
        StringBuffer str = (StringBuffer) obj;
        System.out.println(str.charAt(0));
    }
}

        为了避免出现这种异常,我们可以使用 instanceof 运算符进行判断。

向下转型中使用 instanceof

public class TestCasting3 {
    public static void main(String[] args) {
        Object obj = new String("阿虎学Java");
        if(obj instanceof String){
            String str = (String)obj;
            System.out.println(str.charAt(0));
        }else if(obj instanceof StringBuffer){
            StringBuffer str = (StringBuffer) obj;
            System.out.println(str.charAt(0));
        }
    }
}

 final 关键字

final 关键字的作用:

  • 修饰变量: 被他修饰的变量不可改变。一旦赋了初值,就不能被重新赋值。
    final int MAX_HEIGHT = 180;
  • 修饰方法:该方法不可被子类重写。但是可以被重载
    final void eat(){}
  • 修饰类: 修饰的类不能被继承。比如:Math、String 等。
    final class A {}

修饰方法:

修饰类:

抽象方法和抽象类

抽象方法

        使用 abstract 修饰的方法,没有方法体,只有声明。定义的是一种“规范”,就是告诉子 类必须要给抽象方法ᨀ供具体的实现。

抽象类

        包含抽象方法的类就是抽象类。通过 abstract 方法定义规范,然后要求子类必须定义具 体实现。通过抽象类,我们就可以做到严格限制子类的设计,使子类之间更加通用。

抽象类和抽象方法的基本用法

public class TestAbstractClass {
    public static void main(String[] args) {
        Dog a = new Dog();
        a.shout();
        a.seeDoor();
    }
}
//抽象类
abstract class Animal {
    abstract public void shout(); //抽象方法
}
class Dog extends Animal {
    //子类必须实现父类的抽象方法,否则编译错误
    public void shout() {
        System.out.println("汪汪汪!");
    }
    public void seeDoor(){
        System.out.println("看门中....");
    }
}

抽象类的使用要点:

  1. 有抽象方法的类只能定义成抽象类
  2. 抽象类不能实例化,即不能用 new 来实例化抽象类。
  3. 抽象类可以包含属性、方法、构造方法。但是构造方法不能用来 new 实例, 只能用来被子类调用。
  4. 抽象类只能用来被继承。
  5. 抽象方法必须被子类实现。

标签:Java,String,void,多态,面向对象编程,抽象类,方法,public
From: https://blog.csdn.net/m0_48205251/article/details/136854549

相关文章

  • 【设计模式】Java 设计模式之责任链模式(Chain of Responsibility)
    责任链模式(ChainofResponsibility)一、概述责任链模式是一种行为设计模式,它允许请求在对象链中传递。每个对象都有机会处理该请求,并且能将其传递给链中的下一个对象。这种模式为请求创建了一个处理对象的链,并沿着这条链传递该请求,直到有一个对象处理它为止。二、模式结......
  • 【设计模式】Java 设计模式之状态模式(State)
    深入理解状态模式(State)一、概述状态模式是一种行为设计模式,它允许一个对象在其内部状态改变时改变它的行为。对象看起来好像修改了它的类。状态模式把所有的与一个特定的状态相关的行为放到一个类中,并且将请求委托给当前状态对象来执行。在状态模式中,我们创建表示各种状......
  • Java 面向对象编程进阶六(内部类 )
    目录内部类内部类的概念内部类的分类1、非静态内部类(外部类里使用非静态内部类和平时使用其他类没什么不同)2、静态内部类3、匿名内部类4、局部内部类内部类        内部类是一类特殊的类,指的是定义在一个类的内部的类。实际开发中,为了方便的使用外部类的相......
  • Java 面向对象编程进阶七(字符串 String )
    目录字符串StringString基础String类和常量池String类常用的方法String类常用方法一String类常用方法二字符串相等的判断字符串String        String是我们开发中最常用的类,我们不仅要掌握String类常见的方法,对于String的底层实现也需要掌握好......
  • 一文彻底搞懂令人疑惑的Java和JDK的版本命名!
    你对Java的版本号以及JDK的命名真正清楚嘛?比如:Java8JavaSE8.0JDK1.8……知道这些是怎么回事嘛?知道还有个Java2的说法嘛?知道还有以下说法嘛?J2SE1.3J2SE1.4……现在已经6月份了,到了9月份,一个新的长期支持版本,Java17就要发布了,啥?Java版本都到17了?不不不,我一直在用JDK......
  • 基于java+springboot+vue实现的电影院选票系统(文末源码+Lw+ppt)23-467
    摘要时代在飞速进步,每个行业都在努力发展现在先进技术,通过这些先进的技术来提高自己的水平和优势,电影院选票系统当然不能排除在外。电影院选票系统是在实际应用和软件工程的开发原理之上,运用java语言,前台Vue框架以及后台SpringBoot框架进行开发。首先要进行需求分析,分析出电......
  • 基于java+springboot+vue实现的智慧养老院管理系统(文末源码+Lw+ppt)23-490
    摘 要智慧养老院管理系统采用B/S架构,数据库是MySQL。网站的搭建与开发采用了先进的java进行编写,使用了springboot框架。该系统从三个对象:由管理员和家属、护工来对系统进行设计构建。主要功能包括:个人信息修改,对家属信息、护工信息、老人入住、外出报备、退房登记、每月餐饮......
  • 基于java+springboot+vue实现的学生管理系统(文末源码+Lw+ppt)23-486
    摘  要学生管理系统的目的是让使用者可以更方便的将人、设备和场景更立体的连接在一起。能让用户以更科幻的方式使用产品,体验高科技时代带给人们的方便,同时也能让用户体会到与以往常规产品不同的体验风格。与安卓,iOS相比较起来,学生管理系统在流畅性,续航能力,等方方面面都有......
  • 基于java+SpringBoot+Vuel的制造装备物联及生产管理ERP系统设计与实现
    基于java+SpringBoot+Vuel的制造装备物联及生产管理ERP系统设计与实现开发语言:Java数据库:MySQL技术:SpringBoot+MyBatis工具:IDEA/Ecilpse、Navicat、Maven系统展示前台展示系统简介制造装备物联及生产管理ERP系统在对开发工具的选择上也很慎重,为了便于开发实现,选......
  • java 差异删除 差异更新与删除
    publicbooleanrelation(LongprojectId,List<BsMemberEntity>members){ //1)、获取原关联数据 List<ProProjectAuthorEntity>oldList=this.findByProjectId(projectId); List<Long>oldMemberIds=oldList.stream().map(item->{ returnitem......