JAVA中集合map,set,list详解

在JAVA的util包中有两个所有集合的父接口Collection和Map,它们的父子关系:
java.util
+Collection 这个接口extends自 --java.lang.Iterable接口
+List 接口
-ArrayList 类
-LinkedList 类
-Vector 类 此类是实现同步的

+Queue 接口
+不常用,在此不表.

+Set 接口
+SortedSet 接口
-TreeSet 类
-HashSet

+Map 接口
-HashMap 类 (除了不同步和允许使用 null 键/值之外,与 Hashtable 大致相同.)
-Hashtable 类 此类是实现同步的,不允许使用 null 键值
+SortedMap 接口
-TreeMap 类

以下对众多接口和类的简单说明:首先不能不先说一下数组(Array)

一、Array , Arrays

Java所有“存储及随机访问一连串对象”的做法,array是最有效率的一种。

1、
效率高,但容量固定且无法动态改变。
array还有一个缺点是,无法判断其中实际存有多少元素,length只是告诉我们array的容量。

2、Java中有一个Arrays类,专门用来操作array。
arrays中拥有一组static函数,
equals():比较两个array是否相等。array拥有相同元素个数,且所有对应元素两两相等。
fill():将值填入array中。
sort():用来对array进行排序。
binarySearch():在排好序的array中寻找元素。
System.arraycopy():array的复制。


二、Collection , Map

若撰写程序时不知道究竟需要多少对象,需要在空间不足时自动扩增容量,则需要使用容器类库,array不适用。

1、Collection 和 Map 的区别

容器内每个为之所存储的元素个数不同。
Collection类型者,每个位置只有一个元素。
Map类型者,持有 key-value pair,像个小型数据库。

2、Java2容器类类库的用途是“保存对象”,它分为两类,各自旗下的子类关系

Collection
--List:将以特定次序存储元素。所以取出来的顺序可能和放入顺序不同。
--ArrayList / LinkedList / Vector
--Set : 不能含有重复的元素
--HashSet /TreeSet
Map
--HashMap
--HashTable
--TreeMap

 Map----一组成对的“键值对”对象,即其元素是成对的对象,最典型的应用就是数据字典,并且还有其它广泛的应用。另外,Map可以返回其所有键组成的Set和其所有值组成的Collection,或其键值对组成的Set,并且还可以像数组一样扩展多维Map,只要让Map中键值对的每个“值”是一个Map即可。

 Collection下 1.迭代器

  迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构。迭代器通常被称为“轻量级”对象,因为创建它的代价小。

  Java中的Iterator功能比较简单,并且只能单向移动:

  (1) 使用方法iterator()要求容器返回一个Iterator。第一次调用Iterator的next()方法时,它返回序列的第一个元素。注意:iterator()方法是java.lang.Iterable接口,被Collection继承。


  (2) 使用next()获得序列中的下一个元素。

  (3) 使用hasNext()检查序列中是否还有元素。

  (4) 使用remove()将迭代器新返回的元素删除。

  Iterator是Java迭代器最简单的实现,为List设计的ListIterator具有更多的功能,它可以从两个方向遍历List,也可以从List中插入和删除元素。

  2.List的功能方法

  List(interface): 次序是List最重要的特点;它确保维护元素特定的顺序。List为Collection添加了许多方法,使得能够向List中间插入与移除元素(只推荐LinkedList使用)。一个List可以生成ListIterator,使用它可以从两个方向遍历List,也可以从List中间插入和删除元素。

  ArrayList: 由数组实现的List。它允许对元素进行快速随机访问,但是向List中间插入与移除元素的速度很慢。ListIterator只应该用来由后向前遍历ArrayList,而不是用来插入和删除元素,因为这比LinkedList开销要大很多。

  LinkedList: 由列表实现的List。对顺序访问进行了优化,向List中间插入与删除得开销不大,随机访问则相对较慢(可用ArrayList代替)。它具有方法addFirst()、addLast()、getFirst()、getLast()、removeFirst()、removeLast(),这些方法(没有在任何接口或基类中定义过)使得LinkedList可以当作堆栈、队列和双向队列使用。

  3.Set的功能方法

  Set(interface): 存入Set的每个元素必须是唯一的,这也是与List不同的,因为Set不保存重复元素。加入Set的Object必须定义equals()方法以确保对象的唯一性。Set与Collection有完全一样的接口。Set接口不保证维护元素的次序。

  HashSet: HashSet能快速定位一个元素,存入HashSet的对象必须定义hashCode()。

  TreeSet: 保持次序的Set,底层为树结构。使用它可以从Set中提取有序的序列。

  LinkedHashSet: 具有HashSet的查询速度,且内部使用链表维护元素的顺序(插入的次序)。于是在使用迭代器遍历Set时,结果会按元素插入的次序显示。

  HashSet采用散列函数对元素进行排序,这是专门为快速查询而设计的;TreeSet采用红黑树的数据结构进行排序元素;LinkedHashSet内部使用散列以加快查询速度,同时使用链表维护元素的次序,使得看起来元素是以插入的顺序保存的。需要注意的是,生成自己的类时,Set需要维护元素的存储顺序,因此要实现Comparable接口并定义compareTo()方法。

3、其他特征

* List,Set,Map将持有对象一律视为Object型别。
* Collection、List、Set、Map都是接口,不能实例化。
继承自它们的 ArrayList, Vector, HashTable, HashMap是具象class,这些才可被实例化。
* vector容器确切知道它所持有的对象隶属什么型别。vector不进行边界检查。


三、Collections

Collections是针对集合类的一个帮助类。提供了一系列静态方法实现对各种集合的搜索、排序、线程完全化等操作。
相当于对Array进行类似操作的类——Arrays。
如,Collections.max(Collection coll); 取coll中最大的元素。
Collections.sort(List list); 对list中元素排序


四、如何选择?

1、容器类和Array的区别、择取
* 容器类仅能持有对象引用(指向对象的指针),而不是将对象信息copy一份至数列某位置。
* 一旦将对象置入容器内,便损失了该对象的型别信息。

2、
* 在各种Lists中,最好的做法是以ArrayList作为缺省选择。当插入、删除频繁时,使用LinkedList();
Vector总是比ArrayList慢,所以要尽量避免使用。
* 在各种Sets中,HashSet通常优于HashTree(插入、查找)。只有当需要产生一个经过排序的序列,才用TreeSet。
HashTree存在的唯一理由:能够维护其内元素的排序状态。
* 在各种Maps中
HashMap用于快速查找。
* 当元素个数固定,用Array,因为Array效率是最高的。


下面代码附上以供参考

  1. public class TestApp {
  2. public static void main(String[] args) {
  3. //List-->数组
  4. List<String> list = new ArrayList<String>();
  5. list.add("蹇伟");
  6. list.add("Jerval");
  7. list.add("杰威");
  8. Object[] objects = list.toArray();//返回Object数组
  9. System.out.println("objects:"+Arrays.toString(objects));
  10. String[] strings1 = new String[list.size()];
  11. list.toArray(strings1);//将转化后的数组放入已经创建好的对象中
  12. System.out.println("strings1:"+Arrays.toString(strings1));
  13. String[] strings2 = list.toArray(new String[0]);//将转化后的数组赋给新对象
  14. System.out.println("strings2:"+Arrays.toString(strings2));
  15. //数组-->List
  16. String[] ss = {"JJ","KK"};
  17. List<String> list1 = Arrays.asList(ss);
  18. List<String> list2 = Arrays.asList("AAA","BBB");
  19. System.out.println(list1);
  20. System.out.println(list2);
  21. //List-->Set
  22. List<String> list3 = new ArrayList<String>(new HashSet<String>());
  23. //Set-->List
  24. Set<String> set = new HashSet<String>(new ArrayList<String>());
  25. //数组-->Set
  26. String[] strs = {"AA","BB"};
  27. Set<String> set2 = new HashSet<String>(Arrays.asList(strs));
  28. System.out.println(set2);
  29. //Set-->数组
  30. Set<String> set3 = new HashSet<String>(Arrays.asList("PP","OO"));
  31. String[] strSet = new String[set3.size()];
  32. set3.toArray(strSet);
  33. System.out.println(Arrays.toString(strSet));
  34. //Map操作
  35. Map<String, String> map = new HashMap<String, String>();
  36. map.put("YYY", "UUU");
  37. map.put("RRR", "TTT");
  38. // 将键转化为Set
  39. Set<String> mapKeySet = map.keySet();
  40. // 将值转化为Set
  41. Set<String> mapValuesSet = new HashSet<String>(map.values());
  42. // 将值转化为List
  43. List<String> mapValuesList = new ArrayList<String>(map.values());
  44. }
  45. }

java如何遍历map

  1. package test;
  2. import java.util.ArrayList;
  3. import java.util.HashMap;
  4. import java.util.Iterator;
  5. import java.util.List;
  6. import java.util.Map;
  7. import java.util.Set;
  8. public class TestMap {
  9. //循环遍历map的方法
  10. public static void main(String[] args) {
  11. Map<String, Integer> tempMap = new HashMap<String, Integer>();
  12. tempMap.put("a", 1);
  13. tempMap.put("b", 2);
  14. tempMap.put("c", 3);
  15. // JDK1.4中
  16. // 遍历方法一 hashmap entrySet() 遍历
  17. System.out.println("方法一");
  18. Iterator it = tempMap.entrySet().iterator();
  19. while (it.hasNext()) {
  20. Map.Entry entry = (Map.Entry) it.next();
  21. Object key = entry.getKey();
  22. Object value = entry.getValue();
  23. System.out.println("key=" + key + " value=" + value);
  24. }
  25. System.out.println("");
  26. // JDK1.5中,应用新特性For-Each循环
  27. // 遍历方法二
  28. System.out.println("方法二");
  29. for (Map.Entry<String, Integer> entry : tempMap.entrySet()) {
  30. String key = entry.getKey().toString();
  31. String value = entry.getValue().toString();
  32. System.out.println("key=" + key + " value=" + value);
  33. }
  34. System.out.println("");
  35. // 遍历方法三 hashmap keySet() 遍历
  36. System.out.println("方法三");
  37. for (Iterator i = tempMap.keySet().iterator(); i.hasNext();) {
  38. Object obj = i.next();
  39. System.out.println(obj);// 循环输出key
  40. System.out.println("key=" + obj + " value=" + tempMap.get(obj));
  41. }
  42. for (Iterator i = tempMap.values().iterator(); i.hasNext();) {
  43. Object obj = i.next();
  44. System.out.println(obj);// 循环输出value
  45. }
  46. System.out.println("");
  47. // 遍历方法四 treemap keySet()遍历
  48. System.out.println("方法四");
  49. for (Object o : tempMap.keySet()) {
  50. System.out.println("key=" + o + " value=" + tempMap.get(o));
  51. }
  52. System.out.println("11111");
  53. // java如何遍历Map <String, ArrayList> map = new HashMap <String,
  54. // ArrayList>();
  55. System.out
  56. .println("java 遍历Map <String, ArrayList> map = new HashMap <String, ArrayList>();");
  57. Map<String, ArrayList> map = new HashMap<String, ArrayList>();
  58. Set<String> keys = map.keySet();
  59. Iterator<String> iterator = keys.iterator();
  60. while (iterator.hasNext()) {
  61. String key = iterator.next();
  62. ArrayList arrayList = map.get(key);
  63. for (Object o : arrayList) {
  64. System.out.println(o + "遍历过程");
  65. }
  66. }
  67. System.out.println("2222");
  68. Map<String, List> mapList = new HashMap<String, List>();
  69. for (Map.Entry entry : mapList.entrySet()) {
  70. String key = entry.getKey().toString();
  71. List<String> values = (List) entry.getValue();
  72. for (String value : values) {
  73. System.out.println(key + " --> " + value);
  74. }
  75. }
  76. }
  77. }  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: list、vector、mapset 是标准模板库(STL)的容器,分别用于不同的数据存储和访问方式。 1. list(链表)是一个双向链表,可以根据需要在任意位置插入和删除元素。它没有固定大小,可以根据需要动态分配内存。由于是链表结构,所以访问元素的效率较低,但在插入和删除操作上非常高效。 2. vector(动态数组)是一个支持随机访问的连续内存的数组,可以在尾部追加元素,并且能够根据需要动态分配内存。由于是连续内存,所以在访问元素时效率较高,但在插入和删除操作上相对较慢。 3. map(映射)是一个存储键对的关联容器,通过键(key)来访问(value)。它内部实现了一棵红黑树(平衡二叉树),保证了以对数时间复杂度执行查找、插入和删除操作。键是唯一的,可以用于对数据进行有序存储和遍历。 4. set集合)是一个存储唯一元素的容器,内部实现也是基于红黑树。它自动对元素进行排序,并且保证元素的唯一性。可以利用set进行集合运算,如并集、交集、差集等。 这四种容器在不同的场景下有不同的使用方式: - 如果需要经常进行插入和删除操作,并且不关心元素的顺序,则可以选择使用list。 - 如果需要频繁访问元素,不需要进行插入和删除操作,或者只在尾部进行插入和删除,则可以选择使用vector。 - 如果需要按照键来查找和存储数据,则可以选择使用map。 - 如果需要存储唯一元素,并且需要对元素进行排序和集合运算,则可以选择使用set。 当然,具体的选择还会根据实际需求和性能要求进行权衡。 ### 回答2: list、vector、mapset都是C++标准模板库(STL)的容器,有着不同的特点和用法。 1. list(链表)是双向链表的实现,其的元素可以随意插入和删除。它没有随机访问的功能,只能通过迭代器进行跳跃式访问。它适用于需要频繁的插入和删除操作,并且不需要随机访问的场景。 2. vector(向量)是动态数组的实现,支持随机访问。在vector进行插入和删除操作需要移动其他元素,所以效率可能会低于list。但是它具有随机访问的能力,并且在末尾插入和删除元素的效率较高。vector适用于需要频繁的随机访问和在末尾插入删除元素的场景。 3. map(映射)是一种由键和对组成的容器,通过键进行查找和插入操作。map内部通过红黑树实现,所以键对是按照键的顺序来存储的。map的每个键是唯一的,如果插入一个已存在的键,将会覆盖原有的map适用于需要按键进行查找的场景。 4. set集合)是一种由唯一元素组成的容器,元素按照一定的顺序进行存储。set内部也是通过红黑树实现,所以元素是按照一定顺序排列的。set的元素是唯一的,插入重复的元素将会被忽略。set适用于需要维护一组唯一元素并进行一些集合操作(如并集、交集等)的场景。 总结而言,list适用于频繁的插入和删除操作,vector适用于频繁的随机访问和末尾插入删除操作,而mapset适用于按键进行查找或维护一组唯一元素的场景。根据实际需求选择适当的容器可以提高代码的效率。 ### 回答3: list、vector、mapset是C++标准库常用的容器类,它们分别用于不同的存储和访问数据的需求。 1. list(链表): - 使用双向链表实现,支持快速的插入和删除操作; - 不支持随机访问,只能通过迭代器依次访问元素; - 元素的添加和删除不会导致迭代器失效; - 适用于需要频繁插入和删除元素的场景,但对于随机访问的需求较少。 2. vector(动态数组): - 使用动态数组实现,支持随机访问; - 在尾部插入或删除元素的时间复杂度为常数,其他位置的插入和删除操作需要移动部分元素; - 内存分配是连续的,支持快速的随机访问; - 在需要经常随机访问元素的场景下使用较多。 3. map(有序映射): - 使用红黑树实现,基于键对的有序映射; - 插入和查找操作的时间复杂度为对数时间复杂度; - map的元素按照键的顺序进行排列; - 适用于需要进行查找操作的场景,如字典、索引等。 4. set(有序集合): - 使用红黑树实现,表示一个有序的不重复元素集合; - 插入和查找操作的时间复杂度为对数时间复杂度; - set的元素按照从小到大的顺序排列; - 适用于需要维护有序不重复元素的场景。 总结: - list适用于频繁插入和删除元素的场景; - vector适用于需要经常随机访问元素的场景; - map适用于需要查找操作的场景,按照键有序排列; - set适用于需要维护有序不重复元素的场景。 实际使用,根据具体的需求选择合适的容器可以提高程序的效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值