集合(或者叫容器)是Java的核心知识点,它有着很深的深度。咱们这里不会设计多深,仅仅做为了解入门,深刻了解请移步各类集合源码文章。好的,下面正是开始介绍...java
咱们知道,Java是一门面向对象编程语言,这也就意味着程序中存在着大量的对象。这个时候问题就来了,咱们如何很好的存放和操做对象呢?若是你能明白这个问题,那么你就知道了“集合为什么而生”这个问题的答案。面试
总结一句: Java给咱们提供了工具(集合)方便咱们去存放和操做多个Java对象编程
Java集合的目的是方便操做多个对象,同时,它提供一系列的API来供咱们操做。所以,在初学Java集合的时候咱们更多的是学习这些API的用法。设计模式
对Java集合的API使用有必定了解以后,咱们就应该从面向对象的角度去理解它。为何会抽象出多个接口,以及每一个接口有什么特征。数组
咱们能够总结出几个经常使用的实现类,这几个经常使用的实现类咱们必须知道它的数据结构是什么,何时使用这个类。微信
同时,你还须要学习和了解数据结构:数据结构
学完上面的内容后,咱们基本掌握了经常使用集合的数据结构,也就知道了如何选择合适的集合容器去存储咱们的对象。总之,学完常见实现类的数据结构以后,对它们的使用场景也有了更加清晰的认识。框架
Java集合是一个很庞大的知识点,话很少说,上图感觉下:编程语言
Java容器可分为两大类:工具
迭代器是一种设计模式,它是一个对象,它能够遍历并选择序列中的对象,而开发人员不须要了解该序列的底层结构。迭代器一般被称为“轻量级”对象,由于建立它的代价小
咱们能够发现一个特色,上述全部的集合类,除了Map系列的集合,Collection集合都实现了Iterator接口。咱们能够在源码中追溯到集合的顶层接口,好比Collection接口,能够看到它继承的是类Iterable。
它是Java集合的顶层接口(不包括Map系列的集合,Map接口是Map系列集合的顶层接口)
因此除了Map系列的集合,我么都能经过迭代器来对集合中的元素进行遍历。
Map集合的子类能够用keyset()方法转换成Set集合遍历
Iterable迭代器一共4个方法:
hasNext():判断下个迭代器是否还有下一个元素
next():返回下一个元素的值,而且把自身offset移动下一位
remove():这个能够删除用这个迭代器集合中的元素(注意若是删除以后仍是前面得到的迭代器,你会发现原来的迭代器仍是没变,得从新得到删除元素以后的迭代器)
forEachRemaining:1.8的新方法 能够直接遍历迭代器剩下的元素,若是从最开始的话就是遍历全部的迭代器
补充: 还有一个ListIterator接口,它继承了Iterator接口,但只能用于List集合。它是Iterator接口的升级版,里面除了Iterator含有的功能外,还具备一些其余的功能。Iterator遍历集合元素时,只能单向遍历,而ListIterator能够双向进行遍历、添加元素、设置元素
当使用Iterator对集合元素进行迭代时,Iterator并非把集合元素自己传给了迭代变量,而是把集合元素的值传给了迭代变量(就如同参数传递是值传递,基本数据类型传递的是值,引用类型传递的仅仅是对象的引用变量),因此修改迭代变量的值对集合元素自己没有任何影响。 下面的程序演示了这一点:
public class IteratorExample { public static void main(String[] args){ List<String> list =Arrays.asList("java语言","C语言","C++语言"); Iterator<String> iterator = list.iterator(); while(iterator.hasNext()){ String next = iterator.next();//集合元素的值传给了迭代变量,仅仅传递了对象引用。保存的仅仅是指向对象内存空间的地址 next ="修改后的"; System.out.println(next); } System.out.println(list); } }
输出结果:
修改后的 修改后的 修改后的 [java语言, C语言, C++语言]
Collection的做用就是规定了一个集合有哪些基本的操做。 这里主要是插入数据,清空数据,是否包含,是否相等,集合里的数据个数和转化成熟组这几种操做。
接口中定义的方法:
Collection接口是Set、List、Queue的父接口。下面逐一介绍:
Set集合与Collection集合基本相同,没有提供任何额外的方法。实际上Set就是Collection,只是行为略有不一样(Set不容许包含重复元素)。 Set集合不容许包含相同的元素,若是试图把两个相同的元素加入同一个Set集合中,则添加操做失败,add()方法返回false,且新元素不会被加入。
List集合表明一个元素有序、可重复的集合,集合中每一个元素都有其对应的顺序索引。List集合容许使用重复元素,能够经过索引来访问指定位置的集合元素 。List集合默认按元素的添加顺序设置元素的索引,例如第一个添加的元素索引为0,第二个添加的元素索引为1...... List做为Collection接口的子接口,可使用Collection接口里的所有方法。并且因为List是有序集合,所以List集合里增长了一些根据索引来操做集合元素的方法:
除此以外,Java 8还为List接口添加了以下两个默认方法:
Queue用户模拟队列这种数据结构,队列一般是指“先进先出”(FIFO,first-in-first-out)的容器。队列的头部是在队列中存放时间最长的元素,队列的尾部是保存在队列中存放时间最短的元素。新元素插入(offer)到队列的尾部,访问元素(poll)操做会返回队列头部的元素。一般,队列不容许随机访问队列中的元素。
接口中定义的方法:
Map用户保存具备映射关系的数据,所以Map集合里保存着两组数,一组值用户保存Map里的key,另外一组值用户保存Map里的value,key和value均可以是任何引用类型的数据。Map的key不容许重复,即同一个Map对象的任何两个key经过equals方法比较老是返回false。 以下图所描述,key和value之间存在单向一对一关系,即经过指定的key,总能找到惟一的、肯定的value。从Map中取出数据时,只要给出指定的key,就能够取出对应的value。
Map集合包括Map接口以及Map接口的全部实现类。Map集合具备如下特色:
Map接口经常使用的实现类有:HashMap、HashTable、TreeMap。
咱们先看下以下示例了解下Map集合的用法:
package collection; import java.util.*; public class Muster { public static void main(String[] args) { Map<Integer, String> platformMap = new HashMap<>(); platformMap.put(1, "博客园"); platformMap.put(2, "掘金"); platformMap.put(3, "微信公众号"); platformMap.put(4, "我的博客"); // 尝试添加剧复Map platformMap.put(4, "我的博客"); // 获取全部的key Set<Integer> keys = platformMap.keySet(); for (Integer integer : keys) { System.out.println("Key:" + integer + ",Value:" + platformMap.get(integer)); } } }
以上代码的输出结果为:
Key:1,Value:博客园 Key:2,Value:掘金 Key:3,Value:微信公众号 Key:4,Value:我的博客
从日志能够看出,当咱们尝试重加剧复Map时,并无添加成功。
关于Map集合的详细用法,HashMap、HashTable、TreeMap的区别(这里是重点,面试可能问的比较多,这里不展开说明)
1. 与Set集合的关系 若是把Map里的全部key放在一块儿看,它们就组成了一个Set集合(全部的key没有顺序,key与key之间不能重复),实际上Map确实包含了一个keySet()
方法,用户返回Map里全部key组成的Set集合。 2. 与List集合的关系 若是把Map里的全部value放在一块儿来看,它们又很是相似于一个List:元素与元素之间能够重复,每一个元素能够根据索引来查找,只是Map中索引再也不使用整数值,而是以另一个对象做为索引。
接口中定义的方法:
Map中还包括一个内部类Entry,该类封装了一个key-value对。Entry包含以下三个方法:
Map集合最典型的用法就是成对地添加、删除key-value对,而后就是判断该Map中是否包含指定key,是否包含指定value,也能够经过Map提供的keySet()
方法获取全部key组成的集合,而后使用foreach
循环来遍历Map的全部key,根据key便可遍历全部的value。下面程序代码示范Map的一些基本功能:
public class MapTest { public static void main(String[] args){ Day day1 = new Day(1, 2, 3); Day day2 = new Day(2, 3, 4); Map<String,Day> map = new HashMap<String,Day>(); //成对放入key-value对 map.put("第一个", day1); map.put("第二个", day2); //判断是否包含指定的key System.out.println(map.containsKey("第一个")); //判断是否包含指定的value System.out.println(map.containsValue(day1)); //循环遍历 //1.得到Map中全部key组成的set集合 Set<String> keySet = map.keySet(); //2.使用foreach进行遍历 for (String key : keySet) { //根据key得到指定的value System.out.println(map.get(key)); } //根据key来移除key-value对 map.remove("第一个"); System.out.println(map); } }
输出结果:
true true Day [hour=2, minute=3, second=4] Day [hour=1, minute=2, second=3] {第二个=Day [hour=2, minute=3, second=4]}
参考: