深入探讨Java中的协变、逆变、双变与不变性

深入探讨Java中的协变、逆变、双变与不变性

在Java编程语言中,类型系统是确保程序正确性的关键部分。类型系统的一个重要特性是子类型化(Subtyping),它允许在某些上下文中使用一个类型的实例代替另一个类型的实例。Java支持多种子类型化规则,包括协变(Covariant)、逆变(Contravariant)、双变(Bivariant)和不变性(Invariant)。本文将深入探讨这些概念,并结合实例进行说明。

协变(Covariant)

协变是指允许将子类型用作超类型的实例。在Java中,数组支持协变,这意味着可以将一个子类型的数组赋值给一个超类型的数组变量。例如:

Integer[] integers = new Integer[10];
Number[] numbers = integers; // 协变赋值

然而,这种赋值是危险的,因为它可能导致ArrayStoreException。如果尝试将一个Double对象存储到Number数组中,就会发生这种情况:

numbers[0] = new Double(25); // 抛出ArrayStoreException

逆变(Contravariant)

逆变是指允许将超类型用作子类型的实例。Java泛型不支持逆变,因为泛型类型是不可变的,这意味着你不能将具体类型的泛型赋值给其父类型的泛型。

双变(Bivariant)

双变同时支持协变和逆变。Java中没有直接支持双变,但是通过通配符,我们可以在一定程度上模拟双变行为。

不变性(Invariant)

不变性是指不允许任何类型的替换。Java泛型具有不变性,这意味着具体类型的泛型不能被其父类型或子类型替换。例如:

List<Integer> integers = new ArrayList<>();
List<Number> numbers = integers; // 编译错误

Java泛型中的协变与逆变

Java泛型通过通配符? extends? super支持协变和逆变。使用? extends可以实现协变,例如:

List<? extends Number> numbers = new ArrayList<>(); // 协变

使用? super可以实现逆变,例如:

List<? super B> bList = new ArrayList<>(); // 逆变

协变方法返回类型

Java允许子类覆盖方法时返回更具体的类型。这意味着子类方法的返回类型必须是可以替换超类对应方法返回类型的。例如:

class Animal {}
class Dog extends Animal {}

class AnimalService {
    Animal getAnimal() { return new Animal(); }
}

class DogService extends AnimalService {
    Dog getAnimal() { return new Dog(); } // 协变返回类型
}

通过本文的探讨,我们可以看到Java中的类型系统是如何通过协变、逆变、双变和不变性来增强程序的灵活性和安全性的。理解这些概念对于编写正确、健壮的Java程序至关重要。

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

t0_54coder

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值