java三大特性_详解Java三大特性——多态

3ffd1ab487ff13c06f3b1060b5c844fe.png

多态是继封装、继承之后,面向对象的第三大特性

现实事物经常会体现出多种形态,如学生,学生是人的一种,则一个具体的同学张三既是学生也是人,即出现两种形态。

Java作为面向对象的语言,同样可以描述一个事物的多种形态。如Student类继承了Person类,一个Student的对象便既是Student,又是Person。

Java中多态的代码体现在一个子类对象(实现类对象)既可以给这个子类(实现类对象)引用变量赋值,又可以给这个子类(实现类对象)的父类(接口)变量赋值。

如Student类可以为Person类的子类。那么一个Student对象既可以赋值给一个Student类型的引用,也可以赋值给一个Person类型的引用。
最终多态体现为父类引用变量可以指向子类对象。

多态的前提是必须有子父类关系或者类实现接口关系,否则无法完成多态。
在使用多态后的父类引用变量调用方法时,会调用子类重写后的方法。

文字再怎么讲,都不够生动,直接用代码来体现

老爸要喝酒,那今天喝什么酒呢,

public class Wine {
    public void drinkWine(){
        System.out.println("===今天我要喝什么酒呢====");
        Wine();
    }
    public void Wine(){
        System.out.println("===看看俺今天能喝啥子哟====");
    }
}
public class JNC extends Wine {
    /**
     * @desc 子类重载父类方法
     *        父类中不存在该方法,向上转型后,父类是不能引用该方法的
     * @param a
     * @return void
     */
    public void drinkWine(String a){
        System.out.println("======今天我要喝剑南春====");
        Wine();
    }
    /**
     * 子类重写父类方法
     * 指向子类的父类引用调用Wine时,必定是调用该方法
     */
    public void Wine(){
        System.out.println("=====剑南春喝上啦,好开森=====");
    }
}
public class Test {
    public static void main(String[] args) {
        Wine a = new JNC();
        a.drinkWine();
        a.Wine();
        Wine b = new Wine();
        b.drinkWine();
        b.Wine();
        JNC c= new JNC();
        c.drinkWine("qq");
    }
}

f6399a375f9b0d1ec914b2b325139be7.png

先来看看这一段,

Wine a = new JNC();
a.drinkWine();
a.Wine();

子类剑南春中的drinkWine带有参数,而父类中的drinkWine不带有参数,即父类不存在这个方法

运行的时候,调用的是父类的drinkWine,先输出了

===今天我要喝什么酒呢====

之后继续调用Wine方法,这个时候是去了子类中,指向子类的父类引用调用Wine时,必定是调用子类中的方法,于是输出了

=====剑南春喝上啦,好开森=====

上面的Wine和JNC中的方法,都没有带Static,如果加上Static呢,看一下代码和运行的结果

class Wine {
    public static void drinkWine() {
        System.out.println("===今天我要喝什么酒呢====");
        Wine();
    }
    public static void Wine() {
        System.out.println("===看看俺今天能喝啥子哟====");
    }
}
class JNC extends Wine {
    public static void drinkWine(String a) {
        System.out.println("======今天我要喝剑南春====");
        Wine();
    }
    public static void Wine() {
        System.out.println("=====剑南春喝上啦,好开森=====");
    }
}
class Test {
    public static void main(String[] args) {
        Wine a = new JNC();
        a.drinkWine();
        a.Wine();
    }
}

可以看到,静态方法,即使向上转型,也只能调用自己的方法啦

05bf6d12b94fba3cbbc7f462268a5946.png

上面比较的是子类和父类的方法,在非静态方法和静态方法,父类引用子类的方法,非静态方法下可以调用子类同名的构造函数方法,不能调用不一样的构造方法

静态方法中,子类向上转型后,父类引用都不能进行调用子类的方法

下面来给父类和子类一些变量,以及一些方法,方法都是非静态的

父类,定义了一些姓名,年龄,兴趣爱好等的变量

和一些say和hobby的方法

public class Father {
    private String fathername;
    private int fatherage;
    private String fahterhobby;
    public void say() {
        System.out.println("==我是你爸爸真伟大,养你这么大==");
        myhobby();
    }
    public void myhobby() {
        System.out.println("==我是你爸爸真伟大,只要你妈妈==");
    }
    public Father() {
        super();
    }
    public Father(String fathername, int fatherage, String fahterhobby) {
        this.fathername = fathername;
        this.fatherage = fatherage;
        this.fahterhobby = fahterhobby;
    }
    //省略getters and setters
    @Override
        public String toString() {
        return "Father{" +
                        "fathername='" + fathername + ''' +
                ", fatherage=" + fatherage +
                ", fahterhobby='" + fahterhobby + ''' +
                '}';
    }
    public String toString(String fathername, int fatherage, String fahterhobby) {
        return "Father{
            " +
                "fathername='" + fathername + ''' +
                ", fatherage=" + fatherage +
                ", fahterhobby='" + fahterhobby + ''' +
                '}';
    }
}

子类和父类差不多,其实不应该定义一样的变量,虽然名称改了一下

public class Son extends Father {
    public void say(){
        System.out.println("==爸爸我要出去玩===");
    }
    public void say(String s){
        System.out.println("==爸爸我要出去玩===" +s);
    }
    public void myhobby(String aaa){
        System.out.println("==爸爸给我买这个玩具: " + aaa);
    }
    private String sonname;
    private int sonage;
    private String sonhobby;
    public Son(String sonname, int sonage, String sonhobby) {
        this.sonname = sonname;
        this.sonage = sonage;
        this.sonhobby = sonhobby;
    }
    public Son(String fathername, int fatherage, String fahterhobby, String sonname, int sonage, String sonhobby) {
        super(fathername, fatherage, fahterhobby);
        this.sonname = sonname;
        this.sonage = sonage;
        this.sonhobby = sonhobby;
    }
    //省略getters and setters
    @Override
        public String toString(String sonname, int sonage, String sonhobby) {
        return "Son{" +
                        "sonname='" + sonname + ''' +
                ", sonage=" + sonage +
                ", sonhobby='" + sonhobby + ''' +
                '}';
    }
}

测试:

主要测试如下:

子类中,对say()无参数的方法进行了改写,输出内容不一致了,且有自己新创建的say(String s)带有入参的方法

子类对myhobby也新增了,有了入参

子类中对toString方法也带有入参的

实例化子类对象,父类引用,即向上转型了,调用say(),这个方法子类中有; 调用myhobby(), 子类中没有myhobby(), 只有myhobby(String aaa) 然后还要调用toString(), 有参数和无参数的

public class test {
    public static void main(String[] args) {
        Father father = new Son("张三",35,"LOL","张四",5,"Learn");
        System.out.println("实例化一个Son对象,用父亲接收");
        father.say();
        father.myhobby();
        //代码报错
        // father.myhobby("LOL惊奇娃娃");
        System.out.println(father.toString());
        System.out.println(father.toString("张三",35,"LOL"));
        System.out.println("n");
        Son son = new Son("张四",5,"Learn");
        System.out.println("实例化一个Son对象,用Son接收");
        son.say();
        son.say("上海迪士尼");
        son.myhobby();
        son.myhobby("LOL惊奇娃娃");
        System.out.println(son.toString("张四",5,"Learn"));
    }
}

运行结果如下:

bd08c20b0167d588baa74afc1cf2d482.png
  1. 父类引用调用say(), 由于子类中有这个方法,调用的是子类的这个方法;
    调用myhobby();, 由于子类中没有这个方法,调用的是父类的这个方法;
    调用子类中带有参数的方法,father.myhobby("LOL惊奇娃娃");代码直接报错了
    调用toString(),分别是无参数和有参数,因为子类中只有三个有参数的,没有无参数的,就无参数返回的是父类的,有参数返回的是子类的
  2. 子类引用指向子类对象,调用say()无参数的和有参数的,由于子类中都有,都是子类自己的方法进行返回
    调用myobby()无参数的和有参数的,由于子类中没有无参数的,就去爸爸那儿找了找,返回了爸爸的爱好,子类中有带参数的,就返回了子类自己的
    调用toString(3个参数略),就返回了自己的方法,如果调用不带参数的toString(),就是返回一个父亲中的方法了。。。

总结: 父类引用指向子类,调用返回的时候,看看自己家有没有啊,有啊,哦,不管了,先去儿子家找找,儿子有啊,儿子用你家的,儿子没有啊,回家用自己的方法吧

子类引用指向子类,调用返回的时候,先去自己(即儿子)方法中看看,我自己没有呀,去父亲方法中看看吧

上面看的继承中的父子关系是,爸爸有,儿子有,儿子有新的

下面继续看继承,论父子之间的关系之,爸爸没有,儿子有;

员工对象,只有一个mailCheck()方法,定义了一些变量

public class Employee {
    private String name;
    private String address;
    private int number;
    public Employee(String name, String address, int number) {
        //System.out.println("Employee 构造函数");
        this.name = name;
        this.address = address;
        this.number = number;
    }
    public void mailCheck() {
        System.out.println("邮寄支票给: " + this.name + " " + this.address);
    }
    @Override
        public String toString() {
        return "Employee{" +
                        "name='" + name + ''' +
                ", address='" + address + ''' +
                ", number=" + number +
                '}';
    }
    //省略getters and setters

Salary对象继承了父类

public class Salary extends Employee{
    private double yearsalary;
    // 全年工资
    public double getSalary() {
        return yearsalary;
    }
    public void setSalary(double salary) {
        if(salary >= 0.0)
                this.yearsalary = salary;
    }
    public double computePay() {
        System.out.println("计算工资,付给:" + getName());
        return yearsalary/12;
    }
    public Salary(String name, String address, int number, double yearsalary) {
        super(name, address, number);
        setSalary(yearsalary);
    }
    public void mailCheck() {
        System.out.println("Salary 类的 mailCheck 方法 ");
        System.out.println("邮寄支票给:" + getName()+ " ,工资为:" + yearsalary);
    }
}

测试如下

public class Demo {
    public static void main(String [] args) {
        Salary s = new Salary("员工 A", "北京", 3, 360000.00);
        s.mailCheck();
        double sa = s.computePay();
        System.out.println(sa);
        System.out.println("n");
        Employee e = new Salary("员工 B", "上海", 2, 240000.00);
        e.mailCheck();
        double salary = ((Salary) e).computePay();
        System.out.println(salary);
    }
}

c86968d52cd94c7ea1d5d06ac6db180a.png

必要要强转 ((Salary) e).computePay(); 即必须向下转型,父亲引用转化为子类的,再去调用子类的方法

即回答:论父子之间的关系之,爸爸没有,儿子有

本来是爸爸类型的引用,将爸爸类型的引用向下转型,然后调用

问题来啦,我是父亲,我有两个儿子或者多个儿子呢,闹啥啊, 爸爸没有这个方法,儿子们都有呢,咋办咧

public class Animal {
    void eat() {
        System.out.println("Animal");
    }
}
public class Cat extends Animal {
    public void eat() {
        System.out.println("===我是猫咪我要吃鱼");
    }
    public void work() {
        System.out.println("===我是猫咪我负责抓老鼠");
    }
}
public class Dog extends Animal {
    public void eat() {
        System.out.println("====我是小狗我要吃骨头");
    }
    public void work() {
        System.out.println("====我是小狗我负责看家");
    }
}
public class Test {
    public static void main(String[] args) {
        show(new Cat());
        // 以 Cat 对象调用 show 方法
        System.out.println("n");
        show(new Dog());  // 以 Dog 对象调用 show 方法
        System.out.println("n");
        Animal a = new Cat();  // 向上转型
        a.eat();               // 调用的是 Cat 的 eat
        Cat c = (Cat)a;        // 向下转型
        c.work();        // 调用的是 Cat 的 work
    }
    public static void show(Animal animal)  {
        animal.eat();
        // 类型判断
        if (animal instanceof Cat)  {  // 猫做的事情
            Cat c = (Cat)animal;
            c.work();
        } else if (animal instanceof Dog) { // 狗做的事情
            Dog c = (Dog)animal;
            c.work();
        }
    }
}

00cbf604f509bdc1ee94a65825a93ed1.png

不如咱直接将动物类改成抽象类吧,

public abstract class Animal {
    abstract void eat() ;
}
public class Cat extends Animal {
    public void eat() {
        System.out.println("===我是猫咪我要吃鱼");
    }
    public void work() {
        System.out.println("===我是猫咪我负责抓老鼠");
    }
}
public class Dog extends Animal {
    public void eat() {
        System.out.println("====我是小狗我要吃骨头");
    }
    public void work() {
        System.out.println("====我是小狗我负责看家");
    }
}

40fedb11c3e376cc053954c2dee11c7f.png

更多关于Java的技术和资讯可以关注我的专栏:

Java架构筑基​zhuanlan.zhihu.com
2ca72924127a965d48f7d1b70cb55237.png

专栏免费给大家分享Java架构的学习资料和视频

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
清楚说明了java三大特性:   抽象:处理各种问题的第一步,把具体的问题与解决问题的方法结合起来,这就是抽象的过程.    Java常见的访问控制修饰符:    Public :到处都可以访问,不受包和父子类关系的限制    默认: 只在同包内可以访问,不受父子类关系限制    Protected: 不同包的子类和同包的类中访问,这是一种继承访问。    Private:只有自己类内部能访问,其他部分无论包内外,父子类都不能访问    Final:修饰属性:常量,值不能被修改     要么在声明时候赋值,要么在构造方法中赋值,两次机只能选取一次。       修饰方法 :final方法子类不能将其重写         修饰类:该类不能有子类    Static:修饰属性:静态属性是类所有实例共有,可以用”类名.属性”的方式访问,对于数值型变量,系统给定初始值,不强制要求赋初始值.     方法:可以用”类名.属性”的方式访问,方法也为类的所有实例所共有     代码块 : 加载的优先级最高,甚至比主方法的加载还要早.    对于由final static共同修饰的变量,赋值的机有两个,一是在声明的时候,二是在静态代码块中,两次机有且只能有一次被选择.     继承:    允许存在父子类的关系,一个子类有且只有一个直接的父类(Java中的继承是类之间的单继承.) 当在内存形成类的信息绑定的时候不出现多个交叉点,结构的清晰程度较高.Java本身的运行效率较低,只能达到C语言的五分之一.Java语言舍弃效率的完美要求,换来学习和使用的简单. 属性的遮盖: Father f = new Son(); 属性访问到的是父类的,方法访问到的是子类的. 方法的重载 :在类的内部一种方法出现多种表达方式,如参数表的不同,构造方法的重载 方法的重写:发生在父子类之间,子类方法对父类的方法有新的实现方法,但两个方法定义部分完全一致. 继承的目的: 实现代码的复用.理顺类型对象之间的结构关系. 构建一个父类的时候,要求他的子类与父类必须有理论上的确实的继承关系 代码复用的方法:    继承复用,完全的将父类非私有部分作为子类的内容,若需变化,子类只要重写父类的同名方法即可。 组合复用:选取某个类型中的一个对象的方法实现作为该类的方法实现过程,这两个类也称为理论父类和理论子类. 程序中是不出现extends关键字对两个类关系的描述的.    通过具体实例抽象出父类的过程称为泛化    通过父类来确认子类的过程称为特化.    在代码的类之间关系的设计阶段,理论上要求父子类之间没有相同的同名方法.但语法允许子类重写父类的方法.要求子类当中只有自己特有的方法和属性,其他部分与父类和其他同级子类相同的部分都从父类继承得来.    有关抽象类:    抽象类是用于描述自然界中层级较高的对象类,人们无法对一个抽象类实例化,如动物。只能用一个动物类的子类对抽象类的对象进行实例化。    抽象修饰符abstract 可以修饰类,修饰方法。    抽象类中可以定义非抽象的方法,但如果一个类中有抽象方法,则这个类必须被定义成抽象类。抽象类中有构造方法,但是没有抽象的构造方法。构造方法的存在完全是为了继承关系而存在。    与之形成对比的就是接口,接口中的所有方法要求定义成公开抽象方法,因为这些方法所表示的就是标准,标准的信息必须做到公开化,透明化,例如体育竞赛中的规则。    抽象类与接口之间的对比:接口与抽象类本质上是两个概念,但可以将接口看作是一个抽象层级比抽象类更高的特殊类。由于接口的信息中只有已知不可变值以及抽象方法,所以信息的绑定方式单一,从实现上可以满足继承但也不出现网状的信息绑定效果,不增加访问时的节点数。因此接口允许多继承

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值