泛型疑点

ArrayList是一个使用泛型的类,它的类声明如下:

public class ArrayList<E> extends AbstractList<E> implement List<E> ...{
      public boolean add(E o)...
}

当我们定义一个方法如下:

public void takeAnimal(ArrayList<Animal> animals){
      ...
}

当我定义个实例对象 ArrayList<Animal> animals = new ArrayList<Animal>() 然后使用 takeAnimal(animals) 来调用我们定义的方法,编译并运行程序,程序能够正常运行;
  重新实验,当我创建一个 Animal 的子类 Cat 的实例对象,例如 ArrayList<Cat> cats = new ArrayList<Cat>(),然后我们再使用 takeAnimal(cats) 来调用之前定义的方法,编译,这个时候程序会报错,错误大致如下:

java.util.ArrayList<Animal> cannot be applied to java.utils.arrayList<Dog>

** Cat 明明为 Animal 的子类,为什么不能作为参数传递呢?**
  理由其实简单,根据Java面向对象的特性。试想想,如果在方法内我添加Animal其他子类的对象怎么办,例如:我们添加 Animal 的另一子类 Dog ,代码如下:

public void takeAnimal(ArrayList<Animal> animals){
      animals.add(new Dog);
}

此时 animals 引用实际上持有的是ArrayList<Cat> 对象,而 CatDog 虽然拥有同一父类,但它们之间并没有 is-a 或者 has-a 关系,因此 Dog 不能添加到 animal 中。而在这个方法中 animals.add(new Dog); 确实在语法上没有错误,Java 为了避免这个问题,在编译时就会提示之前的错误。

那么我们是不是要为每一个类的 ArrayList 参数定义一个方法呢?
  答案是不用!!!Java 为我们提供了一个方案:使用泛型与万用字符。例如我们重写方法

public void takeAnimals(ArrayList<? extends Animal> animals){
      ...
}

或者

public <T extends Animal> void takeAnimals(ArrayList<T> animals){
      ...
}

使用上述方法后我们使用 takeAnimal(cats) 来调用方法就不会报错了,但是有一个问题需要注意,在该方法体中不会允许你向 animals 添加任何元素。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 2014年的苹果全球开发者大会(WWDC),当Craig Federighi向全世界宣布“We have new ...
    yeshenlong520阅读 6,809评论 0 9
  • 1.import static是Java 5增加的功能,就是将Import类中的静态方法,可以作为本类的静态方法来...
    XLsn0w阅读 5,013评论 0 2
  • 面向对象主要针对面向过程。 面向过程的基本单元是函数。 什么是对象:EVERYTHING IS OBJECT(万物...
    sinpi阅读 4,768评论 0 4
  • 为什么需要泛型? 通过泛型可以定义类型安全的数据结构,而无须使用实际的数据类型(可扩展)。这能够显著提高性能并得到...
    一只好奇的茂阅读 5,027评论 2 39
  • 以下自定义命令中 *** C:\Program Files\TortoiseSVN\bin\ *** 为 **...
    小歆Pro阅读 5,145评论 0 1