学习Java之如何正确地向上转型与向下转型

 更新时间:2023年05月10日 09:35:45   作者:一一哥Sun  
面向对象的第三个特征是多态,实现多态有三个必要条件:继承、方法重写和向上转型,在学习多态之前,我们还要先学习Java的类型转换,本篇文章就来带大家认识什么是类型转换,看看类型转换都有哪几种情况,以及如何避免类型转换时出现异常

一. 类型转型

将一个类型转换成另一个类型的过程被称为类型转换。 我们所说的对象类型转换,一般是指两个存在继承关系的对象,而不是任意类型的对象。如果两个类型之间没有继承关系,就不允许进行类型转换,否则会抛出强制类型转换异常(java.lang.ClassCastException)。
Java语言允许某个类型的引用变量引用子类的实例,而且可以对这个引用变量进行类型转换。Java中引用类型之间的类型转换(前提是两个类是直接或间接的父子关系)主要有两种,分别是向上转型(upcasting)和向下转型(downcasting)

我们先来看下面这张图:

猫、狗、牛、羊都是动物,所以”动物“是父类,猫狗牛羊是具体的子类。我们可以说猫是动物,狗是动物,牛是动物,羊是动物,这个结论肯定没错!但如果我们说,动物是猫,动物是狗,动物是牛,动物是羊,这个结论一定正确吗?那可就不一定了。子类肯定符合父类类型,但反之,父类不一定符合子类类型

接下来分别对向上转型和向下转型进行讲解。

二. 向上转型

1. 概念

所谓的向上转型,就是父类引用指向子类的对象,也就是把子类对象直接赋给父类引用变量,此时不用强制转换。比如我们说猫是动物,狗是动物,牛是动物,羊是动物,这就是把子类当成父类来用。父类是子类的上级,我们直接把子类向上提拔转型了。

2. 特点

向上转型具有如下特点:

  • 编译类型取决于=号左边,运行类型取决于=号右边;
  • 子类可以调用父类的所有成员,但需遵守访问权限;
  • 父类不能调用子类的特有成员;
  • 最终的运行效果取决于子类的具体实现。

3. 语法

向上转型的基本语法如下所示:

//左侧是父类引用变量,右侧是创建的子类对象,我们可以把它当作父类使用

父类类型 引用名 = new 子类类型();

4. 案例

接下来我们通过一个案例来演示向上转型该如何实现。

/**
 * @author 一一哥Sun 
 * 定义父类---向上与向下转型
 */
public interface Animal {
    //叫
    public void speak();
}

//Dog类实现Animal接口
public class Dog implements Animal {
    @Override
    public void speak() {
	System.out.println("狗子:汪汪");
    }
}

//Cat类实现Animal接口
public class Cat implements Animal {
    @Override
    public void speak() {
	System.out.println("猫子:喵喵");
    }
}

//定义一个测试类
public class AnimalTest {
    public static void main(String[] args) {
	//向上转型
	//父类引用dog变量,指向子类对象new Dog();
        //动物 = 狗;以下代码的意思就是”狗是动物“
	Animal dog=new Dog();
        //向上转型后,可以使用父类Animal中的属性和方法。
	dog.speak();

	//Animal animal=new Cat();效果等同于如下两行代码:
	Cat cat=new Cat();
	Animal animal=cat;
	animal.speak();
    }
}

我们在进行向上转型时,可以写成Animal animal=new Cat(); Animal animal=cat;的形式。就是先创建出cat对象,然后再将cat对象赋值给Animal类型的变量,这样就实现了向上转型。因为Animal是父级类型,Cat是子级类型,Cat是Animal的子类,所以可以直接将cat变量赋值给animal,这就是向上转型。但这种写法比较繁琐,我们通常是直接采用Animal cat=new Cat();的形式,简洁明了。

我们之所以可以实现向上转型,主要是因为两个类型之间是父子关系。比如在本例中,向上转型就等于说”猫是动物,狗是动物“,因为猫狗都是动物的子类,所以这个结论是确定无误的。

三. 向下转型

1. 概念

向下转型则反之,也就是一个已经向上转型的子类对象指向父类引用。 向下转型后,可以调用子类类型中所有的成员。向下转型时,必须进行强制类型转换。因为父类拥有的成员,子类肯定会有,但子类拥有的成员,父类不一定有。

但要注意,如果父类的引用对象指向的是子类对象,则向下转型时是安全的,即编译时不会出现错误。但如果父类的引用对象是父类本身,那么在向下转型的过程中是不安全的,虽然编译时不会出错,但在运行过程中会出现强制类型转换异常。我们可以使用instanceof运算符来避免出现强制类型转换异常。

2. 特点

向下转型具有如下特点:

  • 只能强制转换父类的引用,不能强制转换父类的对象;
  • 父类的引用必须指向子类目标类型的对象;
  • 向下转型后,父类可以调用子类类型中的所有成员。

3. 语法

向下转型的基本语法如下所示:

//向下转型使用强制类型转换的格式,将父类引用类型转换为子类引用类型

子类类型 引用名 = (子类类型) 父类引用;

4. 案例

接下来我们在上面的案例基础之上,对代码进行改造,在Cat类中增加一个新的方法。

public class Cat implements Animal {
    @Override
    public void speak() {
	System.out.println("猫子:喵喵");
    }

    //给其他动物打招呼
    public void helloAnimal(Animal animal) {
	//向下转型,将父类型转为子类型。
	//此时有可能会出现ClassCastException类型转换异常,因为子类一定属于父类的一员,但父类不一定属于子类。
	//我们说“猫是动物”一定没问题,但如果说“动物是猫”,这个结论未必正确。所以把“动物强转成猫”的过程中,有可能会出现异常。
	//只有两者之间具有明确的父子关系时才能进行强转,否则就会出现类型转换异常。
		
	//向下转型时需要进行类型强转
	Cat cat=(Cat) animal;
        //向下转型后,可以使用子类Cat中的属性和方法。
	cat.speak();
    }
}

//测试类
public class AnimalTest {
    public static void main(String[] args) {
	Animal animal=new Cat();
	animal.speak();
		
	Cat cat=new Cat();
        //这里我们传参时,既可以传递animal,也可以传递cat,但不能传递dog类型,否则会出现java.lang.ClassCastException: 
	//因为class com.yyg.convert.Dog cannot be cast to class com.yyg.convert.Cat。狗不能被转成猫
	//cat2.helloAnimal(dog);
	cat.helloAnimal(animal);

        //Dog dog = new Dog();
	//这里就会编译出错,不允许把Dog对象类型转换成Cat对象类型
	//Cat cat = (Cat)dog;    
    }
}

向下转型就是将父类型转为子类型,但要注意,此时有可能会出现ClassCastException类型转换异常,因为子类一定属于父类的一员,但父类不一定属于子类。我们说“猫是动物”一定没问题,但如果说“动物是猫”,这个结论未必正确。所以把“动物强转成猫”的过程中,有可能会出现异常。只有两者之间具有明确的父子关系时才能进行强转,否则就会出现类型转换异常。

就比如上面的案例中,我们传参时,既可以传递animal,也可以传递cat,但不能传递dog类型,否则会出现java.lang.ClassCastException,因为class com.yyg.convert.Dog cannot be cast to class com.yyg.convert.Cat,狗类不能被转成猫类。

四. 结语

至此,就把类型转换给大家介绍完了,我们来看看类型转换的要点吧::

  • 向上转型是父类引用指向子类的对象,不必强制类型转换 
  • 向下转型是子类对象指向父类引用,必须进行强制类型转换;
  • 可以强制向下转型最好借助instanceOf进行类型判断;

以上就是学习Java之如何正确地向上转型与向下转型的详细内容,更多关于Java向上转型与向下转型的资料请关注脚本之家其它相关文章!

相关文章

  • Java开发如何把数据库里的未付款订单改成已付款

    Java开发如何把数据库里的未付款订单改成已付款

    这篇文章主要介绍了Java开发如何把数据库里的未付款订单改成已付款,先介绍MD5算法,简单的来说,MD5能把任意大小、长度的数据转换成固定长度的一串字符,实现思路非常简单需要的朋友可以参考下
    2022-11-11
  • Spring Boot自动配置源码实例解析

    Spring Boot自动配置源码实例解析

    Spring Boot作为Java领域最为流行的快速开发框架之一,其核心特性之一就是其强大的自动配置机制,下面这篇文章主要给大家介绍了关于Spring Boot自动配置源码的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-08-08
  • Java二维数组与稀疏数组相互转换实现详解

    Java二维数组与稀疏数组相互转换实现详解

    在某些应用场景中需要大量的二维数组来进行数据存储,但是二维数组中却有着大量的无用的位置占据着内存空间,稀疏数组就是为了优化二维数组,节省内存空间
    2022-09-09
  • java实现消息队列的两种方式(小结)

    java实现消息队列的两种方式(小结)

    本文主要介绍了两种java实现消息队列的方式,利用Spring消息模板发送消息和Apache ActiveMQ官方实例发送消息,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-12-12
  • java实现播放背景音乐

    java实现播放背景音乐

    这篇文章主要为大家详细介绍了java实现播放背景音乐,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-03-03
  • SpringBoot-JWT生成Token和拦截器的使用(访问受限资源)

    SpringBoot-JWT生成Token和拦截器的使用(访问受限资源)

    本文主要介绍了SpringBoot-JWT生成Token和拦截器的使用(访问受限资源),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • Spring Boot实现简单的定时任务

    Spring Boot实现简单的定时任务

    这篇文章主要给大家介绍了关于利用Spring Boot实现简单的定时任务的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者使用Spring Boot具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2020-07-07
  • Java设计实现一个针对各种类型的缓存

    Java设计实现一个针对各种类型的缓存

    这篇文章主要为大家详细介绍了Java如何设计实现一个针对各种类型的缓存,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-11-11
  • Spring jcl及spring core源码深度解析

    Spring jcl及spring core源码深度解析

    这篇文章主要为大家介绍了Spring jcl及spring core源码深度解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • 如何用java实现分页查询

    如何用java实现分页查询

    这篇文章主要介绍了如何用java实现分页查询,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-06-06

最新评论