重庆分公司,新征程启航

为企业提供网站建设、域名注册、服务器等服务

java泛型容器Collection怎么用

这篇文章主要介绍“java泛型容器Collection怎么用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java泛型容器Collection怎么用”文章能帮助大家解决问题。

创新互联公司基于成都重庆香港及美国等地区分布式IDC机房数据中心构建的电信大带宽,联通大带宽,移动大带宽,多线BGP大带宽租用,是为众多客户提供专业四川移动机房托管报价,主机托管价格性价比高,为金融证券行业服务器托管,ai人工智能服务器托管提供bgp线路100M独享,G口带宽及机柜租用的专业成都idc公司。

先简单来段例子:

public void testGenerics() {
   Collection numbers = new ArrayList<>();
   numbers.add(1); // ok
   numbers.add(0.1); // ok

   Collection numbers2 = new ArrayList<>();
   // don't work, you don't know which subtype 'numbers2' exactly contains
   numbers2.add(1); // oops!
}

这个例子其实有点反人类,估计大部分人(包括我)对这种转换的第一反应肯定是“当然是对的”(这就掉坑了),说下我的理解:

  • Collection:表示这个Collection里包含的都是Number类型的对象,可以是Integer/Long/Float,因为编译器可以判断obj instanceof Number == true;

  • Collection:表示这个Collection是Number类型的“某个子类型”的Collection实例,可以是Collection/Collection,所以调用numbers2.add(1)是不行的,因为编译器不知道这个numbers2包含的元素到底是Number的哪个子类型,编译器无法判断obj instanceof UnknownType的结果;

  • Collection,这个E类型是“一个”具体的类型,而不能是表示某个parent的多种子类型的占位符;

再来个例子:

public void testGenerics() {
   Collection numbers = new ArrayList<>();
   Collection integers = new ArrayList<>();
   Collection numbers2 = new ArrayList<>();
   
   numbers2 = integers; // ok
   numbers2 = numbers; // ok
   
   // don't work, Collection != Collection
   numbers = integers; // oops!
}

Integer明明继承了Number,那为什么

  • Collection == Collection

不成立呢,我们再来看个例子:

public void testGenerics() {
   Collection profits = new ArrayList<>();
   
   insertSomething(profits); // line 1
   
   Integer profit = profits.iterator().next(); // oops! crash
}

private void insertSomething(Collection numbers) {
   numbers.add(Long.MAX_VALUE);
}

如果line 1成立,那么接下去获取利润将会得到个负数,后续的一系列计算都会发声异常,如果代码不够健壮甚至可能会抛出一些意料之外的RuntimeException,导致方法不正常结束甚至程序crash。

所以一句话,Collection != Collection是为了运行期的安全,将可能发生的类型转换异常在编译期就解决掉。

现在再来说说Collection与Collection,又是很多人(包括我)第一反应肯定是“Object是所有java对象的公共父类,所以Collection可以表示任意类型的集合”,来看个例子:

public void testGenerics2() {
   Collection integers = new ArrayList<>();

   Collection objects2 = integers; // ok
   // don't work, which type of 'objects2' contains is uncertain
   objects2.add(1); // oops!
   
   Collection objects = integers; // oops!
}
  • Collection表示的范围比Collection大;

  • 无法调用objects2.add(1)是因为编译器无法精确推断objects2到底是哪种数据类型的容器,可能会产生运行时的类型转换异常;

  • 表示任意数据类型集合的正确写法是Collection

  • Collection不能表示任意类型的集合。

    为什么Collection不是表示任意类型的集合呢,其实也是编译器认为这里有类型转换错误的风险:

    public void testGenerics() {
       Collection integers = new ArrayList<>();

       Collection objects = integers; // oops!
       // don't work, which type of 'objects2' contains is uncertain
       objects.add("1");

       Integer one = objects.iterator().next(); // oops! crash
    }