向上转型概念明细 【原理】:实际就是创建一个子类对象,将其当成父类对象来使用。 语法格式:父类类型对象名new子类类型()AnimalanimalnewCat(元宝,2);animal是父类类型,但可以引用一个子类对象,因为是从小范围向大范围的转换复制代码以下是定义的父类Animal类它的两个继承类Cat类和Dog类,分别重写了Animal类的eat()方法classAnimal{Stringname;intage;publicAnimal(Stringname,intage){this。namename;this。ageage;}publicvoideat(){System。out。println(age岁的name正在吃东西);}}classCatextendsAnimal{publicCat(Stringname,intage){super(name,age);}Overridepublicvoideat(){System。out。println(age岁的name正在吃猫粮);}}classDogextendsAnimal{publicDog(Stringname,intage){super(name,age);}Overridepublicvoideat(){System。out。println(age岁的name正在吃狗粮);}}复制代码 使用场景1:直接赋值以下这种就是直接赋值类型的向上转型,将子类对象的引用给到父类对象。然后再将这个对象作为参数传递进去,就可以根据不同引用调用不同的行为publicclassTest1{publicstaticvoideat(Animalanimal){animal。eat();}publicstaticvoidmain(String〔〕args){Animalanimal1newAnimal(动物,3);Animalanimal2newCat(加菲猫,2);Animalanimal3newDog(哈士奇,1);eat(animal1);eat(animal2);eat(animal3);}}复制代码 使用场景2:方法传参第二种方法传参很简单,我们连对象都不需要创建,只需要直接将三种不同对象的引用作为实参传递给到eat()中的形参接受,就可以产生【向上转型】publicclassTest1{publicstaticvoideat(Animalanimal){animal。eat();}publicstaticvoidmain(String〔〕args){eat(newAnimal(动物,3));eat(newCat(加菲猫,2));eat(newDog(哈士奇,1));}}复制代码使用场景3:方法返回第三种便是在一个方法中将不同对象的引用进行返回,然后传递给到父类对象,也可以产生向上转型publicstaticvoideat(Animalanimal){animal。eat();}复制代码稍微讲一下这个方法,可以看到形参是String字符类型,在方法中通过判断外界传入进来的字符串然后对应地进行返回若是猫就返Cat对象的引用;若是狗就返Dog对象的引用;若是其他就返nullpublicstaticAnimalBuyAnimal(Stringvar){if(var。equals(猫)){returnnewCat(加菲猫,2);}elseif(var。equals(狗)){returnnewDog(哈士奇,1);}else{returnnull;}}复制代码Animalanimal1BuyAnimal(猫);eat(animal1);Animalanimal2BuyAnimal(狗);eat(animal2);Animalanimal3BuyAnimal(兔);eat(animal3);复制代码然后可以看到,前两个传入的是【猫】和【狗】,因此执行的便是不同对象的行为,但是在最后传入了【兔】,因此在BuyAnimal()方法中会returnnull这就使得传入eat()方法中去调用相关对象的行为时因为不存在这个对象的引用,所以就会产生空指针异常 上述便是向上转型的三种场景。学会辨别和使用即可向上转型的优缺点 【优点】:让代码实现更简单灵活,这个从上面的三个不同场景就可以看得出来【缺点】:不能调用到子类特有的方法比如说我在Dog类中写了一个它自己独有的方法lookdoor()classDogextendsAnimal{publicDog(Stringname,intage){super(name,age);}Overridepublicvoideat(){System。out。println(age岁的name正在吃狗粮);}publicvoidlookdoor(){System。out。println(name在看门);}}复制代码但是可以看到,我在通过向上转型之后通过父类对象接受子类引用但是调用不到子类中特有的方法,这其实就是向上转型的缺陷 【注意事项】上转型对象不可以访问子类新增的成员变量和子类自己新增的方法,因为这是子类独有的,而父类中没有上转型对象可以访问子类继承的方法或者是子类重写的方法,这个时候当上转型对象去调用这个方法时,一定是调用了子类重写后的方法,这就是我们前面在讲继承的时候所提到的方法重写不可以将父类创建的对象的引用赋值给子类声明的对象,也就是下面的这两句代码,这很明显和我们的上转型对象相反的,我们是将子类对象给到父类的引用,但这是将父类的引用给到子类的对象,完全就是颠倒黑白【就和猫是动物,动物却不是猫一个道理】向下转型 接下去我们来讲讲向下转型,这种类型的转化不太安全,如果没有特殊需求,不建议使用向下转型解决【调用子类独有方法】继续上面的谈到的在向上转型之后无法调用子类独有的方法这个问题,其实向下转型就可以解决,代码如下Animalanimal1newDog(哈士奇,1);animal1。eat();animal1。lookdoor();Dogdog(Dog)animal1;dog。lookdoor();复制代码可以看到,也就是将父类的对象animal强转成为Dog狗类的对象,这样的话其实就可以去调用子类中特有的方法了 向下转型的缺陷 那为什么说向下转型不安全呢,因为它存在安全隐患Animalanimal1newCat(加菲猫,2);animal1。eat();animal1。lookdoor();Dogdog(Dog)animal1;dog。lookdoor();复制代码可以看到我将原先的Dog狗类变为了Cat猫类,此时animal就得到了猫类对象的引用,但是在下面可以看到如果将这个animal强转为Dog狗类的对象其实就会出问题了看到编译器报出了ClassCastException类型转换异常 向下转型的优化 向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。Java中为了提高向下转型的安全性,引入了instanceof,如果该表达式为true,则可以安全转换可以看到,在进行强转之前我通过instanceof这个关键字进行了一下判断,看看animal1是否获取到了Dog类的引用,若是才可以进行强制类型转化,若不是的话就不会进行任何操作Animalanimal1newCat(加菲猫,2);animal1。eat();animal1。lookdoor();if(animal1instanceofDog){判断一下animal1是否获取到了Dog类的引用Dogdog(Dog)animal1;dog。lookdoor();}复制代码 如果对instance关键字感兴趣的可以看看官方的文档instanceof关键字 再度对比二者【碎碎念】 可能在上面这一系列说来有点难分辨,我们再来对比看看其实你可以这么去想Animalanimal1newCat可以看成是猫归属于一个动物类,那猫一定是属于动物的但是看到Dogdog(Dog)animal1其实就要去思考把动物归属于狗,这其实是说不通的,难道只要是动物就一定是狗吗?那可不一定,动物可多的是 总结与提炼 来总结一下本文所学到的内容在本文中我们讲到了多态中的【向上转型】与【向下转型】首先是说到了向上转型,介绍了它会出现的三种场景,也分析了它的优缺点,知道了在向下转型之后无法调用子类特有的方法但是在向下转型中,我们解决了这个问题,通过对父类对象进行一个强转,就可以调用到子类当中的方法,不过可以看到这种做法不太安全,若是一开始父类对象接受了一个子类的引用,但是在强转的时候转化为了另外一个子类,就会造成类型转换的问题于是后面对方法进行了修正,在前面加上了instanceof关键字进行一个判断,只有父类接收到了这个子类的引用,才可以强转为这个子类的对象 原文链接:https:juejin。cnpost7205406121678225445