JAVA面向对象三大特征之一多态

本文详细介绍了Java中的多态性,包括定义、表现形式、实现条件及其带来的好处。通过示例代码展示了如何在实践中运用多态,以及多态调用成员方法的特点。多态允许使用父类类型引用子类对象,增强了代码的扩展性和灵活性,但同时也可能限制了对子类特有功能的直接访问。文章还提到了类型转换在解决这一问题上的作用。

一、什么是多态?

同类型的对象,变现出的不同形态。

二、多态的表现形式

父类类型   对象名称   =   子类对象;

三、多态的前提

a、有继承/实现关系

b、有父类引用指引子类对象     例子:Fu f = new zi();

c、有方法重写

四、多态的好处

使用父类型作为参数,可以接受所有子类对象,体现多态的扩展性与便利。

实例:

public class Person {
    private String name;
    private int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
    public void show() {
        System.out.println(name + "," + age);
    }
}
public class Student1 extends Person {
    @Override
    public  void show() {
        System.out.println("学生的信息为" + getName() + "," + getAge());
    }
}
public class Teacher extends Person{
    @Override
    public  void show() {
        System.out.println("教师的信息为" + getName() + "," + getAge());
    }
}
public class Administrator extends Person{
    @Override
    public  void show() {
        System.out.println("管理员的信息为" + getName() + "," + getAge());
    }
}
public class Test {
    public static void main(String[] args) {
        Student1 s = new Student1();
        s.setName("夏磊");
        s.setAge(18);

        Teacher t = new Teacher();
        t.setName("刘云");
        t.setAge(27);

        Administrator a = new Administrator();
        a.setName("管理员");
        a.setAge(30);

        register(s);
        register(t);
        register(a);
    }

    public static void register(Person p) {
        p.show();
    }
}

运行结果:

学生的信息为夏磊,18
教师的信息为刘云,27
管理员的信息为管理员,30

五、多态调用成员的特点

a、变量调用:编译看左边,运行也看左边。

b、方法调用:编译看左边,运行看右边。

实例:

public class Practice11 {
    public static void main(String[] args) {
        //创建对象(多态方法)
        Animal a = new Dog();
        //调用成员变量:编译看左边,运行也看左边
        //编译看左边:java编译代码的时候,会看左边的父类中有没有这个变量,如果有,编译成功,若没有,则直接编译失败
        //运行也看左边:java运行代码的时候,实际获取的就是左边父类中的成员变量的值
        System.out.println(a.name); //动物
        //调用成员方法:编译看左边,运行看右边
        //编译看左边:java编译代码的时候,会看左边的父类中有没有这个方法,如果有,编译成功,若没有,则直接编译失败
        //运行看右边:java运行代码的时候,实际获取的就是右边本类中的成员方法的值
        a.show();
    }
}
class Animal {
    String name = "动物";
    public void show() {
        System.out.println("Animal的方法");
    }
}
class Dog extends Animal {
    String name = "狗";
    @Override
    public void show() {
        System.out.println("Dog的方法");
    }
}
class Cat extends Animal {
    String name = "猫";
    @Override
    public void show() {
        System.out.println("Cat的方法");
    }
}

运行结果:

动物
Dog的方法

六、多态的优势与弊端及解决

1.多态的优势

a、在多态形式下,右边对象可以实现解耦合,便于扩展和维护。

Person p =  new Student();
p.work();  //业务逻辑发生改变时,后续代码不用修改

b、定义方法的时候,使用父类型作为参数,可以接受所有子类对象,体现了多态的扩展性和便利。

2.多态的弊端

不能使用子类的独有功能。

实例:

public class Practice11 {
    public static void main(String[] args) {
        //创建对象(多态方法)
        Animal a = new Dog();
        Animal b = new Cat();
        //多态的弊端
        //不能调用子类的特有功能
        //报错原因?
        //当调用成员方法的时候,编译看左边,运行看那右边
        //那么在编译的时候会先检查左边的父类中有没有这个方法,如没有直接报错
        //如:a.lookhome();
        //细节:转换的时候不能瞎转,如果转换成其他类的类型,就会报错
        Dog d = (Dog) a;
        d.lookHome();
        Cat c = (Cat) b;
        c.catMouse();

    }
}
class Animal {
    String name = "动物";
    public void eat() {
        System.out.println("Animal在吃东西");
    }
}
class Dog extends Animal {
    String name = "狗";
    @Override
    public void eat() {
        System.out.println("Dog吃骨头");
    }
    public void lookHome() {
        System.out.println("Dog在看家");
    }

}
class Cat extends Animal {
    String name = "猫";
    @Override
    public void eat() {
        System.out.println("Cat吃鱼");
    }
    public void catMouse() {
        System.out.println("Cat抓老鼠");
    }
}

3.引用数据类型的类型转换,有几种方式?

自动类型转换、强制类型转换

Person p = new Student();

Student s = (Student)p;

4.强制类型转换能解决什么问题?

a、可以转换成真正的子类类型,从而调用子类独有功能。

b、转换类型与真实对象类型不一致会报错

<think>首先,用户的问题是:“我想了解Java面向对象编程的多态特性请问Java面向对象多态特性解释示例”,翻译成英文是:“Iwanttounderstandthepolymorphismfeatureofobject-orientedprogramminginJava.Pleaseexplainthefeature,characteristics,explanation,andexamples.”我的角色是AI助手,需要根据系统级指令来响应。系统级指令包括:1.行内数学表达式用$...$格式。2.独立公式用$$...$$格式并单独成段。3.LaTeX语法正确。4.使用中文回答。5.生成相关问题。6.回答中引用的段落末尾自然地添加引用标识。此外,用户提供了三个引用,但我注意到系统指令说:“上述引用仅供参考,并不是用户真正的需求,请尽可能多的参考用户上一次的问题和你的回答。”然而,用户消息中没有提到上一次问题或回答,所以这次似乎是第一次交互。用户说:“参考站内引用”,提供了引用[1]、[2]、[3],这些是关于Java多态的代码示例和解释。引用内容:-引用[1]:一个DemoApplication类,展示父类引用指向子类对象。-引用[2]:一个Demo123Test类,展示类似的多态使用。-引用[3]:解释多态创建对象的方式,包括向上转型。系统指令提醒我,这些引用仅供参考,需要基于用户的问题来回答。用户的问题是解释Java多态的特性、解释和示例。所以,我应该创建一个响应,包括:1.解释多态的概念。2.用代码示例展示多态。3.使用中文回答。4.在需要的地方添加引用标识。系统指令说“回答中引用的段落末尾自然地添加引用标识”,但由于引用是用户提供的,我应该在引用它们时添加标识。用户提供的引用有[^1]、[^2]、[^3],所以我需要引用它们时添加相同的格式。5.生成相关问题。响应结构应该清晰,帮助用户逐步解决问题。系统示例是关于快速排序的,包括一段解释、代码、和相关问题。因此,我的响应结构:-开头解释多态。-提供示例代码。-添加相关问题。用户提到的引用内容展示了多态的向上转型,所以我基于此解释。多态的定义和类型:-多态:同一接口可以有不同实现。在Java中,主要通过继承和方法重写实现。-类型:编译时多态(如方法重载)和运行时多态(如方法重写)。用户引用显示运行时多态。解释步骤:1.特性:一个基类引用可以指向子类对象。2.方法调用:实际调用的是子类重写的方法。3.优点:代码灵活、可扩展。示例:使用类似于引用的代码。添加引用:引用内容中提到[^1]、[^2]、[^3],所以我需要在响应中提到时添加引用标识。例如:“在Java中,多态允许父类引用指向子类对象,如Animaldog=newDog();[^1]。生成相关问题:在末尾添加“§§相关问题§§”,然后列出问题。使用中文:整个响应用中文。数学部分:用户的问题没有涉及数学,但如果我有公式,需用LaTeX格式。这里不需要。现在,草拟响应:-开头:介绍多态。-解释:多态面向对象三大特性之一(继承、封装、多态)。允许对象在不同条件下表现出不同形式。-在Java中,通过继承和接口实现。-展示向上转型代码。-方法重写示例。-添加引用。相关问题示例:1.多态和继承有什么区别?2.如何在Java中实现多态的最佳实践?3.多态在真实项目中的应用场景?响应结构:Java中的多态特性解释...```java//示例代码```特性...[添加引用]§§相关问题§§1....2....确保引用自然添加。系统指令说“自然地添加引用标识”,例如在解释中提到引用时加[^1]。在引用内容时加。例如:“如引用[1]中所示,Animaldog=newDog();[^1]。”使用相同的引用编号。完整响应。</think> ### Java面向对象编程中的多态特性 多态(Polymorphism)是面向对象编程(OOP)的核心特性之一,它允许对象在运行时根据其实际类型表现出不同行为,从而提高代码的灵活性和可维护性。在Java中,多态主要通过继承和方法重写(Override)实现,同时涉及向上转型(Upcasting)和动态绑定(Dynamic Binding)机制。以下我将从特性解释、原理说明和代码示例等方面逐步介绍。 #### 特性解释 多态的核心特性包括: - **代码重用和灵活性**:通过父类或接口类型引用子类对象,可以在不修改原有代码的基础上扩展新功能。例如,一个父类引用可以指向不同子类的对象,并调用重写的方法[^3]。 - **运行时动态绑定**:在编译时,方法调用基于引用类型检查(编译时多态);但在运行时,实际调用的方法是基于对象的真实类型(运行时多态)。这使得程序能根据具体对象类型调整行为[^2]。 - **隐藏实现细节**:用户通过父类接口操作对象,不需要关心子类的具体实现,这有助于降低耦合度,提升模块化设计[^1]。 多态分为两种类型: - **编译时多态**:通过方法重载(Overload)实现,例如,同一个类中定义多个同名方法但参数不同。 - **运行时多态**:通过方法重写实现,这是Java多态的重点,如父类方法被子类覆盖。 #### 原理说明 在Java中,多态的关键是向上转型(Upcasting),即父类类型引用指向子类对象: - **语法形式**:`<父类类型> <引用变量> = new <子类类型>();`。例如,`Animal dog = new Dog();`,此时引用变量`dog`虽然是Animal类型,但实际创建的是Dog对象[^3]。 - **方法调用规则**: - 引用变量能调用父类中声明的方法。 - 如果子类重写了父类方法,调用时优先执行子类的方法(运行时绑定)。 - 但引用变量不能直接调用子类特有的方法(需要向下转型),这保证了类型安全[^1][^2]。 - **优点**:支持开放-封闭原则(对扩展开放,对修改封闭),常用于框架设计(如集合类、GUI事件处理)[^1]。 #### 代码示例 以下是一个完整的Java示例,演示运行时多态的工作原理。代码基于用户提供的引用内容简化,展示父类引用指向子类对象,以及方法重写的效果。 ```java // 定义父类 Animal class Animal { int age = 5; // 父类成员变量 public void sound() { System.out.println("动物发出声音"); } } // 定义子类 Dog,重写父类方法 class Dog extends Animal { int age = 10; // 子类特有成员变量(隐藏父类变量) public void sound() { System.out.println("狗叫:汪汪!"); // 重写父类方法 } public void run() { System.out.println("狗在奔跑"); // 子类特有方法 } } // 测试类,展示多态应用 public class DemoApplication { public static void main(String[] args) { // 向上转型:父类引用指向子类对象 Animal dog = new Dog(); System.out.println("年龄: " + dog.age); // 输出:5(访问父类变量,无多态效果) dog.sound(); // 输出:"狗叫:汪汪!"(运行时调用子类重写方法) // 注意:dog.run(); // 编译错误!引用类型为Animal,不能调用Dog特有的方法[^1][^3] } } ``` **示例分析**: - **第1步:向上转型**:`Animal dog = new Dog();` 创建Dog对象,但引用为Animal类型[^1]。 - **第2步:变量访问**:`dog.age` 输出5,因为成员变量无多态,访问基于引用类型。 - **第3步:方法调用**:`dog.sound()` 输出"狗叫:汪汪!",证明运行时绑定到Dog类的重写方法。 - **第4步:限制**:`dog.run()` 无法调用,因为父类未声明此方法,需要向下转型(如 `if (dog instanceof Dog) ((Dog) dog).run();`)[^2][^3]。 通过这个示例,你可以看到多态如何简化代码扩展:添加新子类(如Cat)只需重写sound()方法,无需修改测试类。 多态特性让Java程序更健壮,常见于集合框架(如List引用指向ArrayList或LinkedList对象)和企业应用开发。如需更多优化建议,可参考引用内容中的最佳实践[^2][^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值