java 面向对象的特征之三 多态 java instanceof 向上转型 向下转型

本文深入探讨Java中多态性的概念及其应用,包括子类方法重写与重载,向上转型与向下转型,以及如何利用instanceof进行类型检查。通过具体示例,展示了多态性在提高代码复用性和灵活性方面的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 多态性,是面向对象中最重要的概念,在java中有两种体现:

  1. 子类方法的重写(overwrite)和同类方法的重载(overload)
  2. 对象的多态性===可以直接应用在抽象类和接口上
  3. 理解:一个事物的多种表现形式

2. java引用变量的两个类型

  1. 编译时类型:编译时类型由声明该变量的时候使用的类型决定
  2. 运行时类型:运行时类型由实际赋值给变量的对象决定
  • 注:
  • 若编译时类型和运行时类型不一致,就出现了多态性(Polymorphism)

3. 子类对象的多态性的使用前提

  1. 要有类的继承
  2. 子类对父类方法的重写(构造器也是一样的概念)

4. 子类对象的多态性(向上转型)

  1. 父类的 引用指向子类对象 或者是子类对象实体付给父类的引用
  2. 以父类的身份运行子类的实体又称向上转型
  3. 虚拟方法盗用:通过父类引用指向子类的对象实体,当调用方法时,实际是执行子类重写父类的方法
  4. 向上转型肯定是可以转的,可以理解为:女人一定人,但人不一定是女人。
  5. demo
Person p1 = new Man();

5. 程序运行分为编译状态和运行状态

  1. 对于多态性来说,编译的时候,“看左边”,将此引用变量理解为父类的类型
  2. 运行时,“看右边”,关注于真正对象的实体,子类的对象,那么执行的方法就是子类重写的。

6.向下转型(需注意的点)

  1. 格式为 父类实例指向子类引用
  2. 需要用到强制转换,向上转型则为自动转换
  3. 若方法为子类特有的,则编译无法通过,(无法调用,因为引用指向了父类,而父类并没有这个方法)
  4. demo
Person p2 = new Women();
p2.eat();
p2.walk();
//p2.shopping();//编译出错

自动类型转换或者强制类型转换

7. 向下转型正确使用 最好通过 instanceof

  1. 格式:对象a instanceof 类A
  2. 含义:判断对象a是否是类A的一个实例,是的话返回true,否则,返回false。
  3. 若a是A类的实例,那么a也一定是A类的父类的实例
  4. dome
if(p1 instanceof Man) {
	Man m1 = (Man)p1;
	m1.entertainment();
}

if(p1 instanceof Person) {
	System.out.println("你好");
}

8.类的属性无多态性

  1. 父类和子类出现多个同名的属性
  2. 引用是谁,即属性就是谁的属性
  3. demo
Person p1 = new Man();
p1.id;//为父类引用的id属性

9.dome

public class TestPerson {
	public static void main(String[] args) {
		Person p  = new Person();
		p.eat();
		p.walk();
		
		Man m = new Man();
		m.eat();
		m.walk();
		
		
		//子类对象的多态性:父类的 引用指向子类对象   
		//或者是子类对象实体付给父类的引用
		Person p1 = new Man();//以父类的身份运行子类的实体 //向上转型
		//虚拟方法盗用:通过父类引用指向子类的对象实体,当调用方法时,实际是执行子类重写父类的方法
		p1.eat();
		p1.walk();//调用子类的方法
		
		Person p2 = new Women();
		p2.eat();
		p2.walk();
//		p2.shopping(); //无法调用,因为引用指向了父类,而父类并没有这个方法
	
		Women w = (Women)p2;//向下转型,使用强制转换符()
		w.shopping();
		
		//出现异常 :java.lang.ClassCastException
//		Women w1 = (Women)p1;//编译时可以通过,但运行时会报错
//		w1.shopping();
		
		
		
//		Women w2 = (Women)new Man();编译时就出错
		
		//向下转型正确使用 最好通过 instanceof
		if(p1 instanceof Women) {
			Women w1 = (Women)p1;
			w1.shopping();
		}
		//instanceof 
		//格式:对象a instanceof 类A :判断对象a是否是类A的一个实例,是的话返回true,否则,返回false。
		//若a是A类的实例,那么a也一定是A类的父类的实例
		if(p1 instanceof Man) {
			Man m1 = (Man)p1;
			m1.entertainment();
		}
		
		if(p1 instanceof Person) {
			System.out.println("你好");
		}
	}
}
public class Man extends Person {
	private boolean smoking;
	...
	public  void entertainment() {
		System.out.println("男人请客");
	}
}

public class Person {
	private String name;
	private int age;
	...
	public void walk() {
		System.out.println("人走路");
	}
	public void eat() {
		System.out.println("人吃放");
	}
}

public class Women extends Person{
	private boolean isBeauty;
	...
	public void walk() {
		System.out.println("女人人走路");
	}
	public void eat() {
		System.out.println("女人吃放");
	}	
	public void shopping(){
		System.out.println("女人爱购物");
	}
}

10.多态性的应用举例

public class TestAnimal {
	public static void main(String[] args) {
		//没有多态的情况,想调用Dog的方法
		//只能新建func(Dog a)方法
		//但有多态的时候,可以直接test.func(new Dog());,相当于Animal a = new Dog();
		TestAnimal test = new TestAnimal();
		test.func(new Animal());
		test.func(new Dog());
		test.func(new Cat());
	}
	public void func(Animal a) {
		a.eat();
		a.jump();
		
		//若想调用子类特有的方法
		//使用 instanceof
		if(a instanceof Dog) {
			Dog d = (Dog)a;
			d.wang();
		}
		if(a instanceof Cat) {
			Cat c = (Cat)a;
			c.catchMouse();
		}
		
	}
	
	public void func(Dog a) {
		a.eat();
		a.jump();
	}
}
class Animal{
	String name;
	int age;
	public void eat() {
		System.out.println("进食");
	}
	public void jump() {
		System.out.println("跳");
	}
}
class Dog extends Animal{
	
	public void eat() {
		System.out.println("狗吃食");
	}
	public void jump() {
		System.out.println("狗急跳墙");
	}
	public void wang(){
		System.out.println("汪汪叫");
	}
}
class Cat extends Animal{
	public void eat() {
		System.out.println("狗吃鱼");
	}
	public void jump() {
		System.out.println("猫跳");
	}
	public void catchMouse() {
		System.out.println("猫捉老鼠");
	}
}
<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、付费专栏及课程。

余额充值