Java SE 多态
1、多态
方法的多态
//方法重载体现多态
A a = new A();
//这里我们传入不同的参数,就会调用不同sum方法
System.out.println(a.sum(10,20));
System.out.println(a.sum(10,20,30));
//方法重写体现多态
对象的多态
- 一个对象的编译类型和运行类型可以不一致
- 编译类型在定义对象时,就确定了,不能改变
- 运行类型是可以变化的
- 编译类型看定义时 = 号 的左边,运行类型看 = 号的右边
Animal animal = new Dog();//animal编译类型是Animal,运行类型是Dog
animal = new Cat(); //animal 的运行类型变成了 Cat,编译类型仍然是 Animal
多态的向上转型
本质:父类的引用指向了子类的对象
语法:父类类型 引用名 = new 子类类型();
特点:编译类型看左边,运行类型看右边。
可以调用父类中的所有成员(须遵循访问权限)
不能调用子类中特有的成员
最终运行效果看子类(运行类型)的具体实现,
即调用方法时,按照从子类开始查找方法,然后调用,规则按方法的调用规则一致
(从子类找,子类有,则调用,子类无,则在父类找,父类有,并可以调用,则调用,父类无,从祖父类找,祖父类 有,并可以调用,则调用,祖父类无,从Object类找) 如果查找方法的过程中,找到了,但是不能访问,则报错
多态的向下转型
- 语法:子类类型 引用名 = (子类类型)父类引用;
- 只能强转父类的引用,不能强转父类的对象
- 要求父类的引用必须指向的是当前目标类型的对象
- 当向下转型后,可以调用子类类型中所有的成员
Animal animal = new Cat();
Cat cat = (Cat) animal;
属性没有重写之说!属性的值看编译类型
instanceOf 比较操作符,用于判断对象的运行类型是否为XX类型或XX类型的子类型
System.out.println(aa instanceof AA); // true
属性看编译类型,方法看运行类型
Java的动态绑定机制
当调用对象方法的时候,该方法会和该对象的内存地址/运行类型绑定
(子类中没有get方法,调用了父类的get方法,但是父类的get方法中又有一个say方法,这个say方法子类也有,则这个时候调用 子类的say方法)
当调用对象属性时,没有动态绑定机制,哪里声明,哪里使用
多态数组:数组的定义类型为父类类型,里面保存的实际元素类型为子类类型
public class PloyArray {
public static void main(String[] args){ Person[] person = new Person[5];
person[0] = new Person("jack",20);
person[1] = new Student("mary",18,100);
person[2] = new Student("smith",19,30.1);
person[3] = new Teacher("scott",30,20000);
person[4] = new Teacher("king",50,25000); //循环遍历多态数组,调用say
for (int i = 0; i < person.length; i++) {
//person[i] 编译类型是 Person ,运行类型是根据实际情况由JVM来判断
System.out.println(person[i].say()); //动态绑定机制
//类型判断 + 向下转型
if (person[i] instanceof Student){//判断person[i] 的运行类型是不是Student
Student student = (Student)person[i];//向下转型
student.study();
}else if(person[i] instanceof Teacher){
((Teacher)person[i]).teach();
}else if(person[i] instanceof Person){ }else{
System.out.println("你的类型有误,请自己检查...");
}
} }
} public class Person { //父类
private String name;
private int age; 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 String say(){ //返回名字和年龄
return name + "\t" + age;
}
} public class Student extends Person{
private double score; public Student(String name, int age, double score) {
super(name, age);
this.score = score;
} public double getScore() {
return score;
} public void setScore(double score) {
this.score = score;
} //重写父类的say方法
@Override
public String say() {
return "学生 " + super.say() + " score =" + score;
} //特有方法
public void study(){
System.out.println("学生 " + getName() + "正在学习...");
}
} public class Teacher extends Person{
private double salary; public Teacher(String name, int age, double salary) {
super(name, age);
this.salary = salary;
} public double getSalary() {
return salary;
} public void setSalary(double salary) {
this.salary = salary;
} //重写父类的say方法
@Override
public String say() {
return "老师 " + super.say() + " salary =" + salary;
} //特有方法
public void teach(){
System.out.println("老师 " + getName() + "正在授课...");
}
} /*
运行结果
jack 20
学生 mary 18 score =100.0
学生 mary正在学习...
学生 smith 19 score =30.1
学生 smith正在学习...
老师 scott 30 salary =20000.0
老师 scott正在授课...
老师 king 50 salary =25000.0
老师 king正在授课...
*/
多态参数:方法定义的形参类型为父类类型,实参类型允许为子类类型
public class PloyParameter {
public static void main(String[] args) {
Worker tom = new Worker("tom", 2500);
Manager milan = new Manager("milan", 5000, 200000);
PloyParameter ployParameter = new PloyParameter(); ployParameter.showEmpAnnual(tom);
ployParameter.showEmpAnnual(milan); ployParameter.testWork(tom);
ployParameter.testWork(milan); } public void showEmpAnnual(Employee e) { //形式参数为父类类型
System.out.println(e.getAnnual());
} public void testWork(Employee e) {
if (e instanceof Worker) {
((Worker) e).work(); //有向下转型
}else if(e instanceof Manager) {
((Manager) e).manager(); //有向下转型
}else {
System.out.println("不做处理");
}
}
} public class Employee { //父类
private String name;
private double salary; //月工资 public Employee(String name, double salary) {
this.name = name;
this.salary = salary;
} //得到年工资的方法
public double getAnnual() {
return 12 * salary;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public double getSalary() {
return salary;
} public void setSalary(double salary) {
this.salary = salary;
} } public class Worker extends Employee{
public Worker(String name, double salary) {
super(name, salary);
}
public void work() {
System.out.println("普通员工" + getName() + "is working");
} @Override
public double getAnnual() { //普通员工没有其他收入,则直接调用父类方法
return super.getAnnual();
}
} public class Manager extends Employee{ private double bonus; public Manager(String name, double salary, double bonus) {
super(name, salary);
this.bonus = bonus;
} public double getBonus() {
return bonus;
} public void setBonus(double bonus) {
this.bonus = bonus;
} public void manager() {
System.out.println("经理" + getName() + "is managing");
} //重写获取年薪的方法
@Override
public double getAnnual() {
return super.getAnnual() + bonus;
}
} /*
运行结果
30000.0
260000.0
普通员工tomis working
经理milanis managing
*/
Java SE 多态的更多相关文章
- Java SE 简介 & 环境变量的配置
Java SE 简介 & 环境变量的配置 一.Java 技术的三个方向 Java 技术分为三个方向 javaSE( Java Platform Standard Edition 标准版)用来开 ...
- Java Se 基础系列(笔记) -- OO
记录所学到的关于Java Se的一些基础知识 1.对象是通过“属性(成员变量)”和“方法”来分别对应事物所具有的静态属性和动态属性 2.类(Class)是对某一类事物的抽象,对象(Object)为某个 ...
- 黑马程序员 ——Java SE(1)
----<a href="http://www.itheima.com" target="blank">Java培训.Android培训.iOS培训 ...
- Java SE 基础知识(一)
一.基础知识 1. Java SE : Java Standard Edition Java ME : Java Micro Edition Java EE : Java Enterprise Edi ...
- Java SE教程
第0讲 开山篇 读前介绍:本文中如下文本格式是超链接,可以点击跳转 >>超链接<< 我的学习目标:基础要坚如磐石 代码要十份规范 笔记要认真详实 一.java内容介绍 ...
- Java SE 开篇
一. Java SE 开篇 1. Java 基本数据类型及其对应的包装类 基本数据类型 对应的包装类 * byte Byte * boolean Boolean * char Character ...
- JAVA SE回顾及思考(1)——面向对象的特点
学习Java已经三年了,现在开始做Android开发,虽说还在用Java语言但本人现在才真真的意识到无论学什么基础才是最重要的,可能一些刚接触Java或者Android的朋友现在还体会不到基础的重要性 ...
- Java SE(1)
Java SE基础回顾 1.循环语句中的break是终止全部循环,跳出循环体:而continue是终止本次循环,跳执行下一循环 2.return语句有两个作用:返回值:结束方法的运行 3.对于java ...
- Java SE —— 专栏总集篇
前言: Java 语言,是相对于其他语言而言,门槛低,而且功能还强大的一门编程语言,本人十分看好这一门语言,但是,它也是有深度的,看过本人的<数据结构与算法>专栏的同学们有福了,因为本人在 ...
随机推荐
- 【RocketMQ】消息的存储
Broker对消息的处理 BrokerController初始化的过程中,调用registerProcessor方法注册了处理器,在注册处理器的代码中可以看到创建了处理消息发送的处理器对象SendMe ...
- NC23053 月月查华华的手机
NC23053 月月查华华的手机 题目 题目描述 月月和华华一起去吃饭了.期间华华有事出去了一会儿,没有带手机.月月出于人类最单纯的好奇心,打开了华华的手机.哇,她看到了一片的QQ推荐好友,似乎华华还 ...
- VS Code 调教日记(2022.6.26更新)
VS Code 调教日记(2022.6.26更新) 基于msys2的MinGW-w64 GCC的环境配置 下载并安装msys2 到路径...msys2安装路径...\msys64\etc\pacman ...
- RocketMQ 集群的搭建部署 以及rocketmq-console-ng仪表台的安装部署
在 RocketMQ 主要的组件如下. NameServerNameServer 集群,Topic 的路由注册中心,为客户端根据 Topic 提供路由服务,从而引导客户端向 Broker 发送消息.N ...
- 【python】M3U8下载器脚本
[python]M3U8下载器脚本 脚本目标: 1. 输入M3U8文件的链接,得到视频 2.使用异步操作,这样可以快很多,不加锁,因为懒得写,而且影响不大 已知条件: 1.m3u8文件其实就是一个记录 ...
- 集合-list常用方法总结
每个方法使用见下方代码详解 点击查看代码 ArrayList list = new ArrayList(); list.add("AA"); list.add(123); list ...
- jupyter 反向代理配置
抓了下包,看了一下WS连不上,参考这个,问题解决 location / { proxy_pass http://127.0.0.1:8813/; # JUPYTER_PORT 为 Jupyter 运行 ...
- 关于ios的IDFA
了解IDFA,看我这篇文章就够了双11剁手后,我静静的限制了广告追踪 今年双11爆了,据统计,全天交易额1207亿,移动端占比82%,在马云的持续教育和移动端的爆发下,用户在移动端消费的习惯已经不可逆 ...
- hive SQL 初学者题目,实战题目 字符串函数,日期拼接,开窗函数。。。。
sql:Hive实现按照指定格式输出每七天的消费平均数输出格式:2018-06-01~2018-06-07 12.29...2018-08-10~2018-08-16 80.67 答案:-- 1.先将 ...
- 音响音箱/恒温壶/电量显示/电子数字时钟等LED数码管显示驱动IC-VK1640B 8段12位/12段8位显示
市面上最常用的数码管为七段/八段显示,八段数码管比七段数码管多一个发光二极管单元(比七段数码管多一个点),又按能显示多少个"8"可分为1位.2位.4位等等.数码管又分为共阳极驱动/ ...