JavaSE学习笔记(十五)—— 集合概述及Collection接口

一、集合概述

1.1 集合的由来

  我们学习的是面向对象语言,而面向对象语言对事物的描述是通过对象体现的,为了方便对多个对象进行操作,我们就必须把这多个对象进行存储。而要想存储多个对象,就不能是一个基本的变量,而应该是一个容器类型的变量,在我们目前所学过的知识里面,有哪些是容器类型的呢?数组和StringBuffer。但是呢?StringBuffer的结果是一个字符串,不一定满足我们的要求,所以我们只能选择数组,这就是对象数组。而对象数组又不能适应变化的需求,因为数组的长度是固定的,这个时候,为了适应变化的需求,Java就提供了集合类供我们使用。

1.2 数组和集合的区别

【长度区别】

  数组的长度固定
  集合长度可变

【内容不同】

  数组存储的是同一种类型的元素
  而集合可以存储不同类型的元素

【元素的数据类型问题】

  数组可以存储基本数据类型,也可以存储引用数据类型
  集合只能存储引用类型

1.3 集合的继承体系

  刚说过集合是存储多个元素的,但是呢,存储多个元素我们也是有不同需求的:比如说,我要这多个元素中不能有相同的元素,再比如说,我要这多个元素按照某种规则排序一下。针对不同的需求,Java就提供了不同的集合类,这样呢,Java就提供了很多个集合类。这多个集合类的数据结构不同,结构不同不重要的,重要的是你要能够存储东西,并且还要能够使用这些东西,比如说判断,获取等。既然这样,那么,这多个集合类是有共性的内容的,我们把这些集合类的共性内容不断的向上提取,最终就能形成集合的继承体系结构。

  因为Collection是集合继承体系中的根接口,所以我们先来学习Collection。

  Collection 是层次结构中的根接口。Collection 表示一组对象,这些对象也称为 collection 的元素。一些 collection 允许有重复的元素,而另一些则不允许。一些 collection 是有序的,而另一些则是无序的。

二、Collection的基本功能

2.1 添加功能

  • boolean add(Object obj):添加一个元素
  • boolean addAll(Collection c):添加一个集合的元素
public class CollectionDemo {
    public static void main(String[] args) {
        // 创建集合1
        // Collection c1 = new Collection(); //错误,因为接口不能实例化

        Collection c1 = new ArrayList();
        c1.add("abc1");
        c1.add("abc2");
        c1.add("abc3");
        c1.add("abc4");

        // 创建集合2
        Collection c2 = new ArrayList();
        c2.add("abc5");
        c2.add("abc6");
        c2.add("abc7");

        // boolean addAll(Collection c):添加一个集合的元素
        System.out.println("addAll:" + c1.addAll(c2));//true
        System.out.println("c1" + c1);//[abc1, abc2, abc3, abc4, abc5, abc6, abc7]

    }
}

2.2 删除功能

  •  void clear():移除所有元素
  • boolean remove(Object o):移除一个元素
  • boolean removeAll(Collection c):移除一个集合的元素,只要有一个元素被移除了,就返回true。
public class CollectionDemo {
    public static void main(String[] args) {
        // 创建集合1
        Collection c1 = new ArrayList();
        c1.add("abc1");
        c1.add("abc2");
        c1.add("abc3");
        c1.add("abc4");

        // void clear():移除所有元素
        // c1.clear();

        // boolean remove(Object o):移除一个元素
        // System.out.println("remove:" + c1.remove("abc1"));
        // System.out.println("remove:" + c1.remove("abc2"));

        // 创建集合2
        Collection c2 = new ArrayList();
        c2.add("abc2");
        c2.add("abc3");
        c2.add("abc7");

        //boolean removeAll(Collection c):移除一个集合的元素
        //只要有一个元素被移除了,就返回true。
        System.out.println("removeAll:" + c1.removeAll(c2));
        System.out.println("c1:" + c1);//[abc1, abc4]
    }
}

2.3 判断功能

  • boolean contains(Object o):判断集合中是否包含指定的元素
  • boolean containsAll(Collection c):判断集合中是否包含指定的集合元素,只有包含所有的元素,才叫包含
  • boolean isEmpty():判断集合是否为空
public class CollectionDemo {
    public static void main(String[] args) {
        // 创建集合1
        // Collection c1 = new Collection(); //错误,因为接口不能实例化

        Collection c1 = new ArrayList();
        c1.add("abc1");
        c1.add("abc2");
        c1.add("abc3");
        c1.add("abc4");

        // boolean contains(Object o):判断集合中是否包含指定的元素
        // System.out.println("contains:"+c1.contains("abc1"));
        // System.out.println("contains:"+c1.contains("abc2"))

        // boolean isEmpty():判断集合是否为空
        // System.out.println("isEmpty:"+c1.isEmpty());

        // 创建集合2
        Collection c2 = new ArrayList();
        c2.add("abc2");
        c2.add("abc3");
        c2.add("abc7");

        //boolean containsAll(Collection c):判断集合中是否包含指定的集合元素
        //只有包含所有的元素,才叫包含
        System.out.println("containsAll:" + c1.containsAll(c2));//false
        System.out.println("c1:" + c1);
    }
}

2.4 长度功能

  int size():元素的个数

public class CollectionDemo {
    public static void main(String[] args) {// 创建集合对象
        // Collection c = new Collection(); //错误,因为接口不能实例化
        Collection c = new ArrayList();

        c.add("hello");
        c.add("world");
        c.add("java");
   //int size():元素的个数
        System.out.println("size:"+c.size());
    }
}

  注意:

  数组求长度用length属性;

  字符串求长度用length()方法

  集合求长度用size()方法

public class Length_Size {
    public static void main(String[] args) {
        String[] strings = {"aaa","bbb","ccc"};
        String string = "aaabbbccc";
        List<Integer> list = new ArrayList<Integer>();
        list.add(1);
        
        System.out.println("String[].length="+strings.length);
        System.out.println("String.length()="+string.length());
        System.out.println("List.size()="+list.size());
    }
}

2.5 交集功能

  boolean retainAll(Collection c):两个集合都有的元素

public class CollectionDemo {
    public static void main(String[] args) {
        // 创建集合1
        Collection c1 = new ArrayList();
        c1.add("abc1");
        c1.add("abc2");
        c1.add("abc3");
        c1.add("abc4");

        // 创建集合2
        Collection c2 = new ArrayList();
        c2.add("abc2");
        c2.add("abc3");
        c2.add("abc7");

        //boolean retainAll(Collection c):两个集合都有的元素?思考元素去哪了,返回的boolean又是什么意思呢?
        //假设有两个集合A,B。
        //A对B做交集,最终的结果保存在A中,B不变。
        //返回值表示的是A是否发生过改变。
        System.out.println("retainAll:" + c1.retainAll(c2));//true
        System.out.println("c1:" + c1);//[abc2, abc3]
        System.out.println("c2:" + c2);//[abc2, abc3, abc7]
    }
}

三、Collection集合的遍历

3.1 转成数组再遍历

  Object[] toArray():把集合转成数组,可以实现集合的遍历

public class CollectionDemo {
    public static void main(String[] args) {
        // 创建集合对象
        Collection c = new ArrayList();

        // 添加元素
        c.add("hello");
        c.add("world");
        c.add("java");

        // 遍历
        // Object[] toArray():把集合转成数组,可以实现集合的遍历
        Object[] objs = c.toArray();
        for (int x = 0; x < objs.length; x++) {
            // System.out.println(objs[x]);
            // 我知道元素是字符串,我在获取到元素的的同时,还想知道元素的长度。
            // System.out.println(objs[x] + "---" + objs[x].length());
            // 上面的实现不了,原因是Object中没有length()方法
            // 我们要想使用字符串的方法,就必须把元素还原成字符串
            // 向下转型
            String s = (String) objs[x];
            System.out.println(s + "----" + s.length());
        }
    }
}

【练习】

  用集合存储5个学生对象,并把学生对象进行遍历。

/*
 * 分析:
 * A:创建学生类
 * B:创建集合对象
 * C:创建学生对象
 * D:把学生添加到集合
 * E:把集合转成数组
 * F:遍历数组
 */
public class StudentDemo {
    public static void main(String[] args) {
        // 创建集合对象
        Collection c = new ArrayList();

        // 创建学生对象
        Student s1 = new Student("林青霞", 27);
        Student s2 = new Student("风清扬", 30);
        Student s3 = new Student("令狐冲", 33);
        Student s4 = new Student("武鑫", 25);
        Student s5 = new Student("刘晓曲", 22);

        // 把学生添加到集合
        c.add(s1);
        c.add(s2);
        c.add(s3);
        c.add(s4);
        c.add(s5);

        // 把集合转成数组
        Object[] objs = c.toArray();
        // 遍历数组
        for (int x = 0; x < objs.length; x++) {
            // System.out.println(objs[x]);

            Student s = (Student) objs[x];
            System.out.println(s.getName() + "---" + s.getAge());
        }
    }
}

3.2 使用Iterator迭代器遍历(重点)

  Iterator iterator():迭代器,集合的专用遍历方式。

  Iterator接口下有两个方法用于集合的遍历

  • Object next():获取元素,并移动到下一个位置。
  • boolean hasNext():如果仍有元素可以迭代,则返回 true。
public class IteratorDemo {
    public static void main(String[] args) {
        // 创建集合对象
        Collection c = new ArrayList();

        // 创建并添加元素
        c.add("hello");
        c.add("world");
        c.add("java");

        // Iterator iterator():迭代器,集合的专用遍历方式
        Iterator it = c.iterator();// 实际返回的肯定是子类对象,这里是多态

        while (it.hasNext()) {
            String s = (String) it.next();
            System.out.println(s);
        }
    }
}

【迭代器的使用图解】

  

【迭代器的原理解析】

  迭代器为什么不定义成一个类,而是一个接口?

  假设迭代器定义的是一个类,这样我们就可以创建该类的对象,调用该类的方法来实现集合的遍历。但是呢,我们想想,Java中提供了很多的集合类,而这些集合类的数据结构是不同的。所以,存储的方式和遍历的方式应该是不同的,进而它们的遍历方式也应该不是一样的,最终,就没有定义迭代器类的。

  而无论你是哪种集合,你都应该具备获取元素的操作,并且,最好再辅助于判断功能,这样,在获取前先判断,就更不容易出错。也就是说,判断功能和获取功能应该是一个集合遍历所具备的,而每种集合的方式又不太一样,所以我们把这两个功能给提取出来,并不提供具体实现,这种方式就是接口。

  那么,真正的具体实现类在哪里呢?

  在真正的具体的子类中,以内部类的方式体现的。

【迭代器的源码】

public interface Inteator {
    boolean hasNext();
    Object next(); 
}

public interface Iterable {
    Iterator iterator();
}

public interface Collection extends Iterable {
    Iterator iterator();
}

public interface List extends Collection {
    Iterator iterator();
}

public class ArrayList implements List {
    public Iterator iterator() {
        return new Itr();
    }
    
    private class Itr implements Iterator {
        public boolean hasNext() {}
        public Object next(){} 
    }
}


Collection c = new ArrayList();
c.add("hello");
c.add("world");
c.add("java");
Iterator it = c.iterator();     //new Itr();
while(it.hasNext()) {
    String s = (String)it.next();
    System.out.println(s);
}

【练习】

   用集合存储5个学生对象,并把学生对象进行遍历。用迭代器遍历。

public class IteratorTest {
    public static void main(String[] args) {
        // 创建集合对象
        Collection c = new ArrayList();

        // 创建学生对象
        Student s1 = new Student("林青霞", 27);
        Student s2 = new Student("风清扬", 30);
        Student s3 = new Student("令狐冲", 33);
        Student s4 = new Student("武鑫", 25);
        Student s5 = new Student("刘晓曲", 22);

        // 把学生添加到集合中
        c.add(s1);
        c.add(s2);
        c.add(s3);
        c.add(s4);
        c.add(s5);

        // 遍历
        Iterator it = c.iterator();
        while (it.hasNext()) {
            Student s = (Student) it.next();
            System.out.println(s.getName() + "---" + s.getAge());

            // NoSuchElementException 不要多次使用it.next()方法
            // System.out.println(((Student) it.next()).getName() + "---"
            // + ((Student) it.next()).getAge());

        }
        // System.out.println("----------------------------------");

        // for循环改写
        // for(Iterator it = c.iterator();it.hasNext();){
        // Student s = (Student) it.next();
        // System.out.println(s.getName() + "---" + s.getAge());
        // }
    }
}

  注意:不要多次使用it.next()方法,因为每次使用都是访问一个对象。

四、集合的toString()方法源码解析

Collection c = new ArrayList();
c.add("hello");
c.add("world");
c.add("java");

System.out.println(c);

  为什么c输出的不是地址值呢?

  1. Collection c = new ArrayList();
    这是多态,所以输出c的toString()方法,其实是输出ArrayList的toString()
  2. 看ArrayList的toString()
    而我们在ArrayList里面却没有发现toString()。
    以后遇到这种情况,也不要担心,你认为有,它却没有,就应该去它父亲里面看看。
  3. toString()的方法源码
    public String toString() {
        Iterator<E> it = iterator(); //集合本身调用迭代器方法,得到集合迭代器
        if (! it.hasNext())
            return "[]";
    
        StringBuilder sb = new StringBuilder();
        sb.append('[');
        for (;;) {
            E e = it.next(); //e=hello,world,java
            sb.append(e == this ? "(this Collection)" : e);
            if (! it.hasNext())
                //[hello, world, java]
                return sb.append(']').toString();
            sb.append(',').append(' ');
        }
    }

     

posted @ 2019-05-13 11:47  yi0123  阅读(272)  评论(0编辑  收藏  举报