集合体系框架图

集合接口

Java集合类库将接口(interface)与实现(implementation)分离,如上图,Set是一个集合接口,而HashSet与TreeSet都是实现了Set接口的子类。

从上图中也可以看出,集合的基本接口是Collection。Collection<E>接口里有个iterator()方法(来自Iterable接口),该方法返回了一个实现了Iterator接口的对象,可以使用这个迭代器对象依次访问集合中的元素。

既然要迭代访问集合中的元素,那么就必须要考虑到如何获取下一个元素,如何判断已经遍历到结尾,如何移除一个元素等,所以Iterator接口就定义了这三个方法分别对应上述三个问题:next()、hasNext()、remove()等。

使用Iterator遍历的简单方式是for…each。而由集合框架体系图可以知道,Collection扩展了Iterable接口,并且所有集合都实现自Collection接口,所以我们可以知道:标准类库的任何集合都可以使用for…each循环。

使用for each的简单举例如下:

List<String> strs = new ArrayList<String>()

//此处省略,add some params;

for(String str: strs){

//do sth;

}

具体的集合

 

链表

如果从一个数组的中间位置删除一个元素,之后的所有元素都要向前移动一个位置。插入一个元素也是如此(之后的元素统一要向后移动),代价是比较大的。

而链表则不同。链表将每个对象存放在独立的节点中,每个节点还存放着下一个节点的引用,并且链表都是双向链接的(存放着上、下节点的引用)。所以,从链表中删除一个元素,只需要更新所删除元素附近的节点即可。可以想象一下,几个小朋友手拉手,站在中间的小明同学离开了,那么只要小明左边和右边的小朋友重新拉手就好了。

链表是一个有序集合,每个对象的位置比较重要。由于迭代器是描述集合位置的,所以,依赖位置的元素的添加由迭代器来负责。

使用迭代器添加元素只针对有序集合有意义,像Set这种无序集合就没有位置可言,但这些无序集合也都是Iterator的派生类,那么如果Iterator接口定义了add方法,就导致像Set这种集合也必须强制实现Iterator里的这个add方法,因此,Iterator接口里就没有定义add方法,而把add方法定义在一个子接口ListIterator中。所以你再看一下开始的框架体系图,分出来一个ListIterator接口之后,仅仅让有顺序需求的集合去实现它。

链表也有软肋,例如不支持快速随机访问。如果要查看链表中第n个元素,就必须从头开始,越过n-1个元素。使用链表的唯一理由是:尽可能减少在列表中间插入或删除元素所付出的代价。如果列表中只有少数几个元素,用ArrayList就OK了。如果需要对集合进行随机访问,就使用数组或者ArrayList,不要使用链表。

数组列表

有两种访问元素的协议,一种是上文提到的使用迭代器,还有一种是使用get和set来随机访问。数组对于后者很管用。

ArrayList封装了一个动态再分配的对象数组。Vector也是个数组。二者的区别是,Vector的所有方法都是同步的,如果只有一个线程访问Vector,代码要在同步操作上浪费大量时间。而ArrayList不是同步的,因此,在不需要同步时使用ArrayList。

散列集

散列表可以快速查找对象。散列表为每个对象计算一个散列码(hash code,由对象中的实例域产生的一个整数)。不同数据域的对象产生不同的散列码。所以,如果自己定义类,就要负责实现这个类的hashCode方法。但是要注意,自己实现的hashCode方法应该与equals兼容(a.equals(b),则a与b的散列码必须相同)。

在Java中,散列表用链表数组实现,每个链表称为桶(bucket)。(如上图有16个桶)。在查找表中对象的位置时,先计算对象的散列码,然后与桶的总数取余。(散列码 % 桶的总数)。如果计算出的对象所在的桶已经满了,没法放置该对象,这就造成散列冲突(hashcollision)。这时,用新对象与桶中所有对象比较,看该对象是否已存在。如果某个桶快要满了,就要进行再散列,创建一个更大的桶,并将所有元素插入到这个新桶中,丢弃原来的桶。

HashSet实现了基于散列表的集,不关心元素的顺序时,使用HashSet。

树集

TreeSet是个有序集合,以任意顺序将元素插入到集合中,对集合遍历时,每个值将自动按照排序后的顺序呈现。排序是用树结构完成的(如红黑树)。在需要排序时使用TreeSet,不需要排序时使用HashSet。

那么TreeSet是如何排列元素的呢?

TreeSet假定插入的元素都实现了Comparable接口。如果插入自定义对象,就必须通过实现Comparable接口定义排序规则,然后它就会去找这个覆盖的方法查看规则。

但是这样有个灾难性的问题:如果某个对象在某个集合中按A方式排序,在另一个集合中按B方式排序,那就没辙了,能覆盖的方法只有一个,规则只能定一次。

所以,game over?不。TreeSet基于上面的问题,便有了另一种排序方式:使用Comparator接口。该接口声明了一个compare方法。如果按照A方式排序,那么就定义一个实现了Comparator接口的类,将这个类的对象塞到TreeSet的构造器中。可能需要一个例子说明一下:

public static void main(String args[]){

SortedSet<Item> parts = new TreeSet<Item>();

parts.add(new Item("张三","100"));

parts.add(new Item("李四","130"));

parts.add(new Item("王二麻子","120"));

SortedSet<Item> sortByWeight = new TreeSet<Item>(new Comparator<Item>(){

public int compare(Item a,Item b){

String weight1 = a.getWeight();

String weight2 =b.getWeight();

return weight1.compareTo(weight2);

};

});

sortByWeight.addAll(parts);

System.out.println(sortByWeight.toString());

}

队列和双端队列

队列可以让人们有效的在尾部添加一个元素,在头部删除一个元素。双端队列可以在头部、尾部同时添加或者删除元素。

Deque接口由ArrayDeque和LinkedList类实现。这两个类都提供了双端队列,且必要时可以增加队列长度。

映射表

HashMap和TreeMap是映射表的两个通用的实现。HashMap对键进行散列,TreeMap用键的整体排序对元素进行排序,并将其组织成搜索树。

浅谈Java的集合体系的更多相关文章

  1. 浅谈Java的集合框架

    浅谈Java的集合框架 一.    初识集合 重所周知,Java有四大集合框架群,Set.List.Queue和Map.四种集合的关注点不同,Set 关注事物的唯一性,List 关注事物的索引列表,Q ...

  2. 浅谈JAVA集合框架

    浅谈JAVA集合框架 Java提供了数种持有对象的方式,包括语言内置的Array,还有就是utilities中提供的容器类(container classes),又称群集类(collection cl ...

  3. !! 浅谈Java学习方法和后期面试技巧

    浅谈Java学习方法和后期面试技巧 昨天查看3303回复33 部落用户大酋长 下面简单列举一下大家学习java的一个系统知识点的一些介绍 一.java基础部分:java基础的时候,有些知识点是非常重要 ...

  4. 【转】浅谈Java中的hashcode方法(这个demo可以多看看)

    浅谈Java中的hashcode方法 哈希表这个数据结构想必大多数人都不陌生,而且在很多地方都会利用到hash表来提高查找效率.在Java的Object类中有一个方法: public native i ...

  5. 浅谈java类集框架和数据结构(2)

    继续上一篇浅谈java类集框架和数据结构(1)的内容 上一篇博文简介了java类集框架几大常见集合框架,这一篇博文主要分析一些接口特性以及性能优化. 一:List接口 List是最常见的数据结构了,主 ...

  6. 浅谈Java线程安全

    浅谈Java线程安全 - - 2019-04-25    17:37:28 线程安全 Java中的线程安全 按照线程安全的安全程序由强至弱来排序,我们可以将Java语言中各种操作共享的数据分为以下五类 ...

  7. 【转】浅谈Java中的hashcode方法

    哈希表这个数据结构想必大多数人都不陌生,而且在很多地方都会利用到hash表来提高查找效率.在Java的Object类中有一个方法: public native int hashCode(); 根据这个 ...

  8. 浅谈Java中的hashcode方法(转)

    原文链接:http://www.cnblogs.com/dolphin0520/p/3681042.html 浅谈Java中的hashcode方法 哈希表这个数据结构想必大多数人都不陌生,而且在很多地 ...

  9. 浅谈Java中set.map.List的区别

    就学习经验,浅谈Java中的Set,List,Map的区别,对JAVA的集合的理解是想对于数组: 数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型),JAVA集合可以存储和操 ...

随机推荐

  1. apache 添加 ssl_module

    下载httpd2.2.6,以前用http2.2.4会出问题(出现个什么lib的错误) ./configure --prefix=/usr/local/apache --enable-ssl --ena ...

  2. Delphi实例之一个较复杂的记事本的实现

    http://www.mamicode.com/info-detail-110813.html delphi中控件位置及自动排版的问题 http://blog.csdn.net/avan_lau/ar ...

  3. Cocoa 新的依赖管理工具:Carthage

    昨天搞了一下pod的安装 因为之前我都是在使用pod来进行第三方库的管理 但是拿到项目之后 竟发现这个前辈是用Carthage 说真的在这之前我从来没有用过这个玩意因为我感觉用POD已经很好了啊 很方 ...

  4. App Store Review Guideline(带翻译)

    1. Terms and conditions(法律与条款) 1.1  As a developer of applications for the App Store you are bound b ...

  5. WinMerge文件编码设置

    http://blog.sina.com.cn/s/blog_7575fab10101o0na.html 系统默认是System codepage,我们要选用Custom codepage.值得注意的 ...

  6. MD5碰撞后时代,MD5还有存在的意义吗?

    MD5是一种HASH函数,又称杂凑函数,由32位16进制组成,在信息安全范畴有广泛和首要运用的暗码算法,它有类似于指纹的运用.在网络安全协议中, 杂凑函数用来处理电子签名,将冗长的签名文件紧缩为一段一 ...

  7. [CSS3] 学习笔记-CSS动画特效

    在CSS3中,出现了很多出彩的效果,例如2D.3D以及过度.动画和多列等.这些效果为页面设计添加了很多的可选设计. 1.2D.3D转换 转换,是使元素改变尺寸.形状.位置的一种效果:通过CSS3转换, ...

  8. synchronized的使用及注意事项

    主要来源:http://blog.csdn.net/luoweifu/article/details/46613015 1.synchronized(this) void method(){ sync ...

  9. Unity编程标准导引-3.3 Transform

    本文为博主原创文章,欢迎转载.请保留博主链接http://blog.csdn.net/andrewfan 每个游戏对象(GameObject),其存在于游戏世界,都有一个位置.朝向.大小等基本定位信息 ...

  10. 从并发处理谈PHP进程间通信(二)System V IPC

    .container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...