首先来看看容器的作用或者说语言设计者设计容器的初衷,也是从《Java编程思想》一书总结出来的:
使用容器原因:
一句话概括:在任意时刻、任意位置、创建任意数量的对象。
其实就是保存对象,然后其实用数组也能保存对象,只是数组长度限定死,所以才用到动态长度的动态数组:容器。
原因分点:
1.动态创建对象(运行时才决定创建对象)
2.对象的类型和数量也是动态的
来看一个小示例:
这示例主要说明上面提到的几点之中的:运行时才知道要创建多少个对象,不可能用个循环然后里面逐一定义对象引用吧,所以用到了容易,把要创建的对象都放到容器里面去,然后再从中取出来,而且容器也给我们提供了很多好用的方法。
然后这里的代码可以有个小改进就是用foreach循环来遍历容器里面的对象:
我们发现,上面都只是用容易来存储特定类型(Applet类)的对象,容器好比我们生活中的罐子,我们当然不想这个罐子只能装水,我们还想拿它来装酸菜,等等。于是,看下面的示例代码:
该示例试图用该容器装两种不同类型的对象,结果发现装是装地进去,但是取出来呢?如果用跟前面同样的一次遍历,会有一个运行时报异常的情况发生,难道一一拿出来看看是Apple还是Orange再决定该怎么转换类型?哦,对了,忘了说了,这里几个代码都未制定容器存储对象的类型,所以容器默认存储的是Object对象,也就是啥对象都能放进去,只不过引用被进行了类型转型(所谓的上转型)而已。接着刚才的,嗯,不可能你把罐子里的东西都倒出来,然后看是不是都是Apple,去挑那个Orange出来吧。所以解决办法是用“泛型”,对,就是这么一个强大而重要的家伙,对于编程能力进一步提升绕不开的点,几乎各种语言都有。先来看用泛型怎么解决这样的一个问题:限定容器只能存储某一类东西,而不能让容器使用者把别的类型的东西装进来:
该示例代码会在编译时即报错,用了泛型则规定了ArrayList容器只能存储Apple这一类的对象,如果试图装一个Orange对象进去,则直接编译报错。
然后下面是正确地只把Apple装进去的示例代码:
import java.util.ArrayList;
import java.util.Scanner;
/**
* @author lv.lang
*
*/
class Apple
{
private static long counter;
private final long id = counter++;
public long id()
{
return id;
}
}
class Orange{}
public class Collection1 {
public static void main(String[] args) {
ArrayList<Apple> apples = new ArrayList<Apple>();
Scanner sc = new Scanner(System.in);
System.out.println("Enter number:");
int num = sc.nextInt();
for(int i=0; i < num; i++)
apples.add(new Apple());
long id = 0;
/*for(int i=0; i < apples.size(); i++)
{
id = ((Apple)apples.get(i)).id();
System.out.println(id);
}*/
//用foreach更快地遍历容器所有对象(同时自动执行了类型转换)
for(Apple a : apples)
{
System.out.println(a.id());
}
}
}
最后截图自《Java编程思想》一书的内容,解析了容器类的基本概念: