Java中泛型的协变

本文探讨了Java泛型的协变特性,并对比了C#中的实现方式。通过具体示例,展示了Java泛型不支持协变的原因及如何正确处理泛型参数。

在工作中遇到一个问题,用代码描述如下:

package test;
import java.util.LinkedList;
import java.util.List;

public class ListTest {
    public void func(List<Base> list) {
    }
    public static void main(String args[]) {
        ListTest lt = new ListTest();
        List<Derived> list = new LinkedList<Derived>();
        lt.func(list); // 编译报错
    }
}

class Base {
}

class Derived extends Base {
}

这里需要写一个函数func,能够以Base的list作为参数。原以为传一个Derived的list也可以,因为Derived是Base的派生类,那Derived的list也应当是Base的list的派生类,结果编译器报错。

究其原因,在网上查了一些资料:Java的泛型并非协变的。

泛型的协变和逆变都是术语,前者指能够使用比原始指定的派生类型的派生程度更小(不太具体的)的类型,后者指能够使用比原始指定的派生类型的派生程度更大(更具体的)的类型。

例如C#中的泛型就是支持协变的:

IEnumerable<Derived> d = new List<Derived>();
IEnumerable<Base> b = d;

但是Java的泛型却是不支持协变的,类似上面的代码在Java中无法通过编译。

但有趣的是,Java中的数组却是支持协变,例如:

Integer[] intArray = new Integer[10]; 
Number[] numberArray = intArray;

总结:Java的泛型不支持协变,更多的是从类型安全的角度考虑。这种设计不是一定必须的,例如C#就没有采用这种设计。只能说Java的设计者在易用性和类型安全之间做了取舍。

最后回到最初的那个问题,要实现一个那样的方法func,可以修改为:

public void func(List list) {
}

或者采用参数化类型:

public <T> void func(List<T> list) {
}

但是这样也有问题,会模糊了func的参数类型。更好的办法是不改func,在传参时就传一个Base类型的List,这就要求在将元素加入这个List时就要转型成Base类型。

PS:vipyami说的方法也不错,通过限制参数类型:

public void func(List<? extends Base> list) {
}







通过短时倒谱(Cepstrogram)计算进行时-倒频分析研究(Matlab代码实现)内容概要:本文主要介绍了一项关于短时倒谱(Cepstrogram)计算在时-倒频分析中的研究,并提供了相应的Matlab代码实现。通过短时倒谱分析方法,能够有效提取信号在时间与倒频率域的特征,适用于语音、机械振动、生物医学等领域的信号处理与故障诊断。文中阐述了倒谱分析的基本原理、短时倒谱的计算流程及其在实际工程中的应用价值,展示了如何利用Matlab进行时-倒频图的可视化与分析,帮助研究人员深入理解非平稳信号的周期性成分与谐波结构。; 适合人群:具备一定信号处理基础,熟悉Matlab编程,从事电子信息、机械工程、生物医学或通信等相关领域科研工作的研究生、工程师及科研人员。; 使用场景及目标:①掌握倒谱分析与短时倒谱的基本理论及其与傅里叶换的关系;②学习如何用Matlab实现Cepstrogram并应用于实际信号的周期性特征提取与故障诊断;③为语音识别、机械设备状态监测、振动信号分析等研究提供技术支持与方法参考; 阅读建议:建议读者结合提供的Matlab代码进行实践操作,先理解倒谱的基本概念再逐步实现短时倒谱分析,注意参数设置如窗长、重叠率等对结果的影响,同时可将该方法与其他时频分析方法(如STFT、小波换)进行对比,以提升对信号特征的理解能力。
### Java 中的逆 #### 逆 (Contravariance) 在Java中,逆是指允许使用某个的超类来替代该。具体来说,`? super T` 表示可以接受 `T` 类或其任何超类的实例。这一特性主要应用于写入操作,确保能够向集合中添加合适的元素。 例如: ```java List<? super Integer> list = new ArrayList<>(); list.add(1); // 合法, 因为Integer是Number的一个子类 // list.get(0) 返回 Object 类,因为无法确定确切类 ``` 上述代码展示了如何利用逆机制实现更灵活的数据存储方式[^1]。 #### (Covariance) 相对于逆而言,则是指当一个作为另一个的子类时所表现出的行为模式。即对于两个不同的实际参数化类A<T1>, B<T2>, 如果存在继承关系使得T1是T2的子类,则可以说A<T1>也是B<T2>的一种特例形式。在Java里我们通常看到的形式就是`<?> extends T`, 它意味着此位置上的类量既可以是具体的`T`也可以是从`T`派生出来的任意子类。 下面是一个简单的例子说明的应用场景: ```java public class Animal {} public class Dog extends Animal {} List<Dog> dogs = Arrays.asList(new Dog()); List<? extends Animal> animals = dogs; for (Animal animal : animals){ System.out.println(animal); } ``` 这里可以看到由于采用了的方式定义列表animals, 所以可以直接赋值给由Dog组成的列表dogs而不必担心编译错误的发生[^3]. #### 不性(Invariance) 值得注意的是,在某些情况下,既不支持也不支持逆的情况被称为不性。这意味着只有当两个的具体类完全一致的情况下才能互相转换。比如`ArrayList<String>`就不能被当作`ArrayList<Object>`处理,即使String确实是Object的子类之一。 总结起来,通过合理运用这些概念可以帮助开发者编写更加通用且安全高效的程序逻辑结构。同时也能有效减少不必要的强制转所带来的风险以及提高代码可读性和维护便利度。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值