背景
所有集合类都位于java.util包下。集合中只能保存对象(保存对象的引用变量)。当我们把一个对象放入集合中后,系统会把所有集合元素都当成Object类的实例进行处理。
Java的集合类主要由两个接口派生而出:Collection和Map,Collection和Map是Java集合框架的根接口,这两个接口又包含了一些接口或实现类。
Set和List接口是Collection接口派生的两个子接口,Queue是Java提供的队列实现,类似于List。
Map实现类用于保存具有映射关系的数据(key-value)。
Set、List和Map可以看做集合的三大类。
接口 |
实现 |
历史集合类 |
Set |
HashSet |
|
|
TreeSet |
|
List |
ArrayList |
Vector |
|
LinkedList |
Stack |
Map |
HashMap |
Hashtable |
|
TreeMap |
Properties |
Collection
Collection接口支持如添加和除去等基本操作。设法除去一个元素时,如果这个元素存在,除去的仅仅是集合中此元素的一个实例。
containsAll() 方法允许您查找当前集合是否包含了另一个集合的所有元素,即另一个集合是否是当前集合的子集。其余方法是可选的,因为特定的集合可能不支持集合更改。 addAll() 方法确保另一个集合中的所有元素都被添加到当前的集合中,通常称为并。 clear() 方法从当前集合中除去所有元素。 removeAll() 方法类似于 clear() ,但只除去了元素的一个子集。 retainAll() 方法类似于 removeAll() 方法,不过可能感到它所做的与前面正好相反:它从当前集合中除去不属于另一个集合的元素,即交。
import java.util.*; public class CollectionToArray { public static void main(String[] args) { Collection collection1=new ArrayList();//创建一个集合对象 collection1.add("000");//添加对象到Collection集合中 collection1.add("111"); collection1.add("222"); System.out.println("集合collection1的大小:"+collection1.size()); System.out.println("集合collection1的内容:"+collection1); collection1.remove("000");//从集合collection1中移除掉 "000" 这个对象 System.out.println("集合collection1移除 000 后的内容:"+collection1); System.out.println("集合collection1中是否包含000 :"+collection1.contains("000")); System.out.println("集合collection1中是否包含111 :"+collection1.contains("111")); Collection collection2=new ArrayList(); collection2.addAll(collection1);//将collection1 集合中的元素全部都加到collection2中 System.out.println("集合collection2的内容:"+collection2); collection2.clear();//清空集合 collection2 中的元素 System.out.println("集合collection2是否为空 :"+collection2.isEmpty()); //将集合collection1转化为数组 Object s[]= collection1.toArray(); for(int i=0;i<s.length;i++){ System.out.println(s[i]); } } }
运行结果为: 集合collection1的大小:3 集合collection1的内容:[000, 111, 222] 集合collection1移除 000 后的内容:[111, 222] 集合collection1中是否包含000 :false 集合collection1中是否包含111 :true 集合collection2的内容:[111, 222] 集合collection2是否为空 :true 111 222
Iterator
Collection 接口的 iterator() 方法返回一个 Iterator。
Collection books = new HashSet(); books.add("Java jichu"); books.add("Java web"); Iterator it = books.iterator(); while (it.hasNext()) { String book = (String) it.next(); System.out.println(book); if(book.equals("Java web")) { books.remove(book); } } Iterator it2 = books.iterator(); while (it2.hasNext()) { String book = (String) it2.next(); System.out.println(book); }
运行结果:
Java jichu
Java web
Java jichu
1) 使用方法 iterator() 要求容器返回一个Iterator .第一次调用Iterator 的next() 方法时,它返回集合序列的第一个元素。
2)使用next() 获得集合序列的中的下一个元素。
3)使用hasNext()检查序列中是否元素。
4) 使用remove()将迭代器新返回的元素删除
Set
Set最大的特性就是不允许在其中存放的元素是重复的。
Set set1 = new HashSet(); if(set1.add("a")) { System.out.println("OK"); } if (set1.add("a")) { System.out.println("OK2"); } else { System.out.println("BAD2"); } set1.add("000"); set1.add("111"); set1.add("222"); System.out.println("集合set1的内容:"+set1); System.out.println("集合set1的大小:"+set1.size()); set1.remove("000"); System.out.println("集合set1的内容:"+set1); System.out.println("集合set1中是否包含000 :"+set1.contains("000")); System.out.println("集合set1中是否包含111 :"+set1.contains("111")); Set set2=new HashSet(); set2.add("111"); set2.addAll(set1);//将set1 集合中的元素全部都加到set2中 System.out.println("集合set2的内容:"+set2); Iterator iterator = set1.iterator();//得到一个迭代器 while (iterator.hasNext()) {//遍历 String element =(String) iterator.next(); System.out.println("iterator = " + element); } //将集合set1转化为数组 Object s[]= set1.toArray(); for(int i=0;i<s.length;i++){ System.out.println(s[i]); }
运行结果: OK BAD2 集合set1的内容:[222, 111, a, 000] 集合set1的大小:4 集合set1的内容:[222, 111, a] 集合set1中是否包含000 :false 集合set1中是否包含111 :true 集合set2的内容:[222, a, 111] iterator = 222 iterator = 111 iterator = a 222 111 a
HashSet、LinkedHashSet、TreeSet
Set set1 = new HashSet(); Set set2 = new LinkedHashSet(); for(int i =0 ; i<5 ;i++) { int s = (int)(Math.random()*100); set1.add(new Integer(s)); set2.add(new Integer(s)); System.out.println("第 "+i+" 次随机数产生为:"+s); } System.out.println("未排序前HashSet:"+set1); System.out.println("未排序前LinkedHashSet:"+set2); //使用TreeSet来对另外的Set进行重构和排序 Set sortedSet = new TreeSet(set1); System.out.println("排序后 TreeSet :"+sortedSet);
运行结果: 第 0 次随机数产生为:85 第 1 次随机数产生为:41 第 2 次随机数产生为:81 第 3 次随机数产生为:49 第 4 次随机数产生为:99 未排序前HashSet:[85, 49, 81, 99, 41] 未排序前LinkedHashSet:[85, 41, 81, 49, 99] 排序后 TreeSet :[41, 49, 81, 85, 99]
List
当我们不知道存储的数据有多少的情况,我们就可以使用List 来完成存储数据的工作。例如前面提到的一种场景。我们想要在保存一个应用系统当前的在线用户的信息。我们就可以使用一个List来存储。因为List的最大的特点就是能够自动的根据插入的数据量来动态改变容器的大小。在 List 中搜索元素可以从列表的头部或尾部开始,如果找到元素,还将报告元素所在的位置。
LinkedList queue = new LinkedList(); queue.addFirst("Bernadine"); queue.addFirst("Elizabeth"); queue.addFirst("Gene"); queue.addFirst("Elizabeth"); queue.addFirst("Clara"); System.out.println(queue); queue.removeLast(); queue.removeLast(); System.out.println(queue);
运行结果:
[Clara, Elizabeth, Gene, Elizabeth, Bernadine]
[Clara, Elizabeth, Gene]
Stack<String> v = new Stack<String>(); v.push("111"); v.push("222"); v.push("333"); System.out.println(v); System.out.println(v.peek()); System.out.println(v); System.out.println(v.pop()); System.out.println(v);
运行结果: [111, 222, 333] 333 [111, 222, 333] 333 [111, 222]
List list = new ArrayList(); list.add("aaa"); list.add("bbb"); list.add("ccc"); list.add("ddd"); System.out.println("下标0开始:"+list.listIterator(0).next());//next() System.out.println("下标1开始:"+list.listIterator(1).next()); System.out.println("子List 1-3:"+list.subList(1,3));//子列表 ListIterator it = list.listIterator();//默认从下标0开始 //隐式光标属性add操作 ,插入到当前的下标的前面 it.add("sss"); while(it.hasNext()){ System.out.println("next Index="+it.nextIndex()+",Object="+it.next()); } //set属性 ListIterator it1 = list.listIterator(); it1.next(); it1.set("ooo"); ListIterator it2 = list.listIterator(list.size());//下标 while(it2.hasPrevious()){ System.out.println("previous Index="+it2.previousIndex()+",Object="+it2.previous()); }
运行结果: 下标0开始:aaa 下标1开始:bbb 子List 1-3:[bbb, ccc] next Index=1,Object=aaa next Index=2,Object=bbb next Index=3,Object=ccc next Index=4,Object=ddd previous Index=4,Object=ddd previous Index=3,Object=ccc previous Index=2,Object=bbb previous Index=1,Object=aaa previous Index=0,Object=ooo
ListIterator
接口继承 Iterator
接口以支持添加或更改底层集合中的元素,还支持双向访问。
Map
数学中的映射关系在Java中就是通过Map来实现的。它表示,里面存储的元素是一个对(pair),我们通过一个对象,可以在这个映射关系中找到另外一个和这个对象相关的东西。
Object put(Object key,Object
value)
:用来存放一个键
-
值对
Map
中
Object remove(Object
key)
:根据
key(
键
)
,移除一个键
-
值对,并将值返回
void putAll(Map mapping)
:将另外一个
Map
中的元素存入当前的
Map
中
void clear()
:清空当前
Map
中的元素
Object get(Object key)
:根据
key(
键
)
取得对应的值
boolean containsKey(Object key)
:判断
Map
中是否存在某键(
key
)
boolean containsValue(Object
value):
判断
Map
中是否存在某值
(value)
int
size():
返回
Map
中
键-值对的个数boolean isEmpty()
:判断当前
Map
是否为空
public Set keySet()
:返回所有的键(
key
),并使用
Set
容器存放
public Collection values()
:返回所有的值(
Value
),并使用
Collection
存放
public Set entrySet()
:
返回一个实现 Map.Entry
接口的元素 SetHashMap<String, String> hm = new HashMap<String, String>(); //试图将2个key为null的key-value对放入HashMap中 hm.put(null , null); hm.put(null , null); //将一个value为null的key-value对放入HashMap中 hm.put("a" , null); hm.put("a" , "b"); //输出Map对象 System.out.println(hm);
运行结果: {null=null, a=b}
HashMap若遇到重复的key,覆盖掉以前的
Map<String, String> map1 = new HashMap<String, String>(); Map<String, String> map2 = new HashMap<String, String>(); map1.put("1","aaa1"); map1.put("2","bbb2"); map2.put("10","aaaa10"); map2.put("11","bbbb11"); //根据键 "1" 取得值:"aaa1" System.out.println("map1.get(\"1\")="+map1.get("1")); // 根据键 "1" 移除键值对"1"-"aaa1" System.out.println("map1.remove(\"1\")="+map1.remove("1")); System.out.println("map1.get(\"1\")="+map1.get("1")); map1.putAll(map2);//将map2全部元素放入map1中 map2.clear();//清空map2 System.out.println("map1 IsEmpty?="+map1.isEmpty()); System.out.println("map2 IsEmpty?="+map2.isEmpty()); System.out.println("map1 中的键值对的个数size = "+map1.size()); System.out.println("KeySet="+map1.keySet());//set System.out.println("values="+map1.values());//Collection System.out.println("entrySet="+map1.entrySet()); System.out.println("map1 是否包含键:11 = "+map1.containsKey("11")); System.out.println("map1 是否包含值:aaa1 = "+map1.containsValue("aaa1"));
运行结果: map1.get("1")=aaa1 map1.remove("1")=aaa1 map1.get("1")=null map1 IsEmpty?=false map2 IsEmpty?=true map1 中的键值对的个数size = 3 KeySet=[2, 10, 11] values=[bbb2, aaaa10, bbbb11] entrySet=[2=bbb2, 10=aaaa10, 11=bbbb11] map1 是否包含键:11 = true map1 是否包含值:aaa1 = false
Map<Integer, String> map1 = new HashMap<Integer, String>(); Map<Integer, String> map2 = new LinkedHashMap<Integer, String>(); for(int i=0;i<10;i++){ int s=(int)(Math.random()*100);//产生一个随机数,并将其放入Map中 map1.put(new Integer(s),"第 "+i+" 个放入的元素:"+s+"\n"); map2.put(new Integer(s),"第 "+i+" 个放入的元素:"+s+"\n"); } System.out.println("未排序前HashMap:"+map1); System.out.println("未排序前LinkedHashMap:"+map2); //使用TreeMap来对另外的Map进行重构和排序 Map<Integer, String> sortedMap = new TreeMap<Integer, String>(map1); System.out.println("排序后:"+sortedMap); System.out.println("排序后:"+new TreeMap<Integer, String>(map2));
运行结果: 未排序前HashMap:{68=第 3 个放入的元素:68 , 32=第 5 个放入的元素:32 , 71=第 7 个放入的元素:71 , 53=第 9 个放入的元素:53 , 37=第 8 个放入的元素:37 , 42=第 4 个放入的元素:42 , 24=第 2 个放入的元素:24 , 63=第 6 个放入的元素:63 , 29=第 0 个放入的元素:29 , 88=第 1 个放入的元素:88 } 未排序前LinkedHashMap:{29=第 0 个放入的元素:29 , 88=第 1 个放入的元素:88 , 24=第 2 个放入的元素:24 , 68=第 3 个放入的元素:68 , 42=第 4 个放入的元素:42 , 32=第 5 个放入的元素:32 , 63=第 6 个放入的元素:63 , 71=第 7 个放入的元素:71 , 37=第 8 个放入的元素:37 , 53=第 9 个放入的元素:53 } 排序后:{24=第 2 个放入的元素:24 , 29=第 0 个放入的元素:29 , 32=第 5 个放入的元素:32 , 37=第 8 个放入的元素:37 , 42=第 4 个放入的元素:42 , 53=第 9 个放入的元素:53 , 63=第 6 个放入的元素:63 , 68=第 3 个放入的元素:68 , 71=第 7 个放入的元素:71 , 88=第 1 个放入的元素:88 } 排序后:{24=第 2 个放入的元素:24 , 29=第 0 个放入的元素:29 , 32=第 5 个放入的元素:32 , 37=第 8 个放入的元素:37 , 42=第 4 个放入的元素:42 , 53=第 9 个放入的元素:53 , 63=第 6 个放入的元素:63 , 68=第 3 个放入的元素:68 , 71=第 7 个放入的元素:71 , 88=第 1 个放入的元素:88 }
从运行结果,我们可以看出,HashMap的存入顺序和输出顺序无关。而LinkedHashMap 则保留了键值对的存入顺序。TreeMap则是对Map中的元素进行排序。在实际的使用中我们也经常这样做:使用HashMap或者LinkedHashMap 来存放元素,当所有的元素都存放完成后,如果使用则是需要一个经过排序的Map的话,我们再使用TreeMap来重构原来的Map对象。这样做的好处是:因为HashMap和LinkedHashMap 存储数据的速度比直接使用TreeMap 要快,存取效率要高。当完成了所有的元素的存放后,我们再对整个的Map中的元素进行排序。这样可以提高整个程序的运行的效率,缩短执行时间。
转载请注明出处:http://www.cnblogs.com/yydcdut/p/3705054.html
原文:http://www.cnblogs.com/yydcdut/p/3705054.html