Java集合框架 (二)

本文深入探讨Java集合框架的核心组件,包括List、Set及其关键实现如ArrayList、LinkedList、HashSet和TreeSet。文章还介绍了如何利用这些类高效管理数据,以及如何自定义排序方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


2.2.AbstractList AbstractSequentialList 抽象类
有两个抽象的 List 实现类: AbstractList AbstractSequentialList 。像 AbstractSet 类一样,它们覆盖了 equals() hashCode() 方法以确保两个相等的集合返回相同的哈希码。若两个列表大小相等且包含顺序相同的相同元素,则这两个列表相等。这里的 hashCode() 实现在 List 接口定义中指定,而在这里实现。
除了 equals() hashCode() AbstractList AbstractSequentialList 实现了其余 List 方法的一部分。因为数据的随机访问和顺序访问是分别实现的,使得具体列表实现的创建更为容易。需要定义的一套方法取决于您希望支持的行为。您永远不必亲自提供的是 iterator 方法的实现。
2.3. LinkedList 类和 ArrayList
在“集合框架”中有两种常规的 List 实现: ArrayList LinkedList 。使用两种 List 实现的哪一种取决于您特定的需要。如果要支持随机访问,而不必在除尾部的任何位置插入或除去元素,那么, ArrayList 提供了可选的集合。但如果,您要频繁的从列表的中间位置添加和除去元素,而只要顺序的访问列表元素,那么, LinkedList 实现更好。
ArrayList LinkedList 都实现 Cloneable 接口,都提供了两个构造函数,一个无参的,一个接受另一个 Collection
2.3.1. LinkedList
LinkedList 添加了一些处理列表两端元素的方法。
(1)  void addFirst(Object o): 将对象 o 添加到列表的开头
void addLast(Object o) :将对象 o 添加到列表的结尾
(2)  Object getFirst(): 返回列表开头的元素
Object getLast(): 返回列表结尾的元素
(3)  Object removeFirst(): 删除并且返回列表开头的元素
Object removeLast(): 删除并且返回列表结尾的元素
(4)  LinkedList(): 构建一个空的链接列表
LinkedList(Collection c): 构建一个链接列表,并且添加集合 c 的所有元素
『使用这些新方法,您就可以轻松的把 LinkedList 当作一个堆栈、队列或其它面向端点的数据结构。』
2.3.2. ArrayList
ArrayList 封装了一个动态再分配的 Object[] 数组。每个 ArrayList 对象有一个 capacity 。这个 capacity 表示存储列表中元素的数组的容量。当元素添加到 ArrayList 时,它的 capacity 在常量时间内自动增加。
在向一个 ArrayList 对象添加大量元素的程序中,可使用 ensureCapacity 方法增加 capacity 。这可以减少增加重分配的数量。
(1)  void ensureCapacity(int minCapacity): ArrayList 对象容量增加 minCapacity
(2)  void trimToSize(): 整理 ArrayList 对象容量为列表当前大小。程序可使用这个操作减少 ArrayList 对象存储空间。
2.3.2.1. RandomAccess 接口
    一个特征接口。该接口没有任何方法,不过你可以使用该接口来测试某个集合是否支持有效的随机访问。 ArrayList Vector 类用于实现该接口。
3. Set 接口
Set 接口继承 Collection 接口,而且它不允许集合中存在重复项,每个具体的 Set 实现类依赖添加的对象的 equals() 方法来检查独一性。 Set 接口 没有引入新方法,所以 Set 就是一个 Collection ,只不过其行为不同。
3.1. Hash
    Hash 表是一种数据结构,用来查找对象。 Hash 表为每个对象计算出一个整数,称为 Hash Code( 哈希码 ) Hash 表是个链接式列表的阵列。每个列表称为一个 buckets( 哈希表元 ) 。对象位置的计算  index = HashCode % buckets (HashCode 为对象哈希码, buckets 为哈希表元总数 )
当你添加元素时,有时你会遇到已经填充了元素的哈希表元,这种情况称为 Hash Collisions( 哈希冲突 ) 。这时,你必须判断该元素是否已经存在于该哈希表中。
如果哈希码是合理地随机分布的,并且哈希表元的数量足够大,那么哈希冲突的数量就会减少。同时,你也可以通过设定一个初始的哈希表元数量来更好地控制哈希表的运行。初始哈希表元的数量为  buckets = size * 150% + 1 (size 为预期元素的数量 )
如果哈希表中的元素放得太满,就必须进行 rehashing( 再哈希 ) 。再哈希使哈希表元数增倍,并将原有的对象重新导入新的哈希表元中,而原始的哈希表元被删除。 load factor( 加载因子 ) 决定何时要对哈希表进行再哈希。在 Java 编程语言中,加载因子默认值为 0.75 ,默认哈希表元为 101
3.2. Comparable 接口和 Comparator 接口
在“集合框架”中有两种比较接口: Comparable 接口和 Comparator 接口。 String Integer Java 内建类实现 Comparable 接口以提供一定排序方式,但这样只能实现该接口一次。对于那些没有实现 Comparable接口的 类、或者自定义的类,您可以通过 Comparator 接口来定义您自己的比较方式。
3.2.1. Comparable 接口
java.lang 包中, Comparable 接口适用于一个类有自然顺序的时候。假定对象集合是同一类型,该接口允许您把集合排序成自然顺序。
(1) int compareTo(Object o): 比较当前实例对象与对象 o ,如果位于对象 o 之前,返回负值,如果两个对象在排序中位置相同,则返回 0 ,如果位于对象 o 后面,则返回正值
在 Java 2 SDK版本1.4中有二十四个类实现 Comparable 接口。下表展示了8种基本类型的自然排序。虽然一些类共享同一种自然排序,但只有相互可比的类才能排序。
排序
BigDecimal,BigInteger,Byte, Double, Float,Integer,Long,Short
按数字大小排序
Character
按 Unicode 值的数字大小排序
String
按字符串中字符 Unicode 值排序
利用 Comparable 接口创建您自己的类的排序顺序,只是实现 compareTo() 方法的问题。通常就是依赖几个数据成员的自然排序。同时类也应该覆盖 equals() hashCode() 以确保两个相等的对象返回同一个哈希码。
3.2.2. Comparator 接口
若一个类不能用于实现 java.lang.Comparable ,或者您不喜欢缺省的 Comparable 行为并想提供自己的排序顺序(可能多种排序方式),你可以实现 Comparator接口,从而定义一个比较器
(1) int compare(Object o1, Object o2): 对两个对象 o1 o2 进行比较,如果 o1 位于 o2 的前面,则返回负值,如果在排序顺序中认为 o1 o2 是相同的,返回 0 ,如果 o1 位于 o2 的后面,则返回正值
『与 Comparable 相似, 0 返回值不表示元素相等。一个 0 返回值只是表示两个对象排在同一位置。由 Comparator 用户决定如何处理。如果两个不相等的元素比较的结果为零,您首先应该确信那就是您要的结果,然后记录行为。』
(2) boolean equals(Object obj): 指示对象 obj 是否和比较器相等。
该方法覆写 Object equals() 方法,检查的是 Comparator 实现的等同性,不是处于比较状态下的对象。
3.3. SortedSet 接口
“集合框架”提供了个特殊的 Set 接口: SortedSet 它保持元素的有序顺序。 SortedSet 接口为集的视图 ( 子集 ) 和它的两端(即头和尾)提供了访问方法。当您处理列表的子集时,更改视图会反映到源集。此外,更改源集也会反映在子集上。发生这种情况的原因在于视图由两端的元素而不是下标元素指定,所以如果您想要一个特殊的高端元素 toElement 在子集中,您必须找到下一个元素。
    添加到 SortedSet 实现类 的元素必须实现 Comparable 接口 ,否则您必须给它的构造函数提供一个 Comparator 接口的实现。 TreeSet 类是它的唯一一份实现。
『因为集必须包含唯一的项,如果添加元素时比较两个元素导致了 0 返回值(通过 Comparable compareTo() 方法或 Comparator compare() 方法),那么新元素就没有添加进去。如果两个元素相等,那还好。但如果它们不相等的话,您接下来就应该修改比较方法,让比较方法和 equals() 的效果一致。』
(1)  Comparator comparator(): 返回对元素进行排序时使用的比较器,如果使用 Comparable 接口的 compareTo() 方法对元素进行比较,则返回 null
(2)  Object first(): 返回有序集合中第一个 ( 最低 ) 元素
(3)  Object last(): 返回有序集合中最后一个 ( 最高 ) 元素
(4)  SortedSet subSet(Object fromElement, Object toElement): 返回从 fromElement( 包括 ) toElement( 不包括 ) 范围内元素的 SortedSet 视图 ( 子集 )
(5) SortedSet headSet(Object toElement): 返回 SortedSet 的一个视图,其内各元素皆小于 toElement
(6) SortedSet tailSet(Object fromElement): 返回 SortedSet 的一个视图,其内各元素皆大于或等于 fromElement
3.4. AbstractSet 抽象类
AbstractSet 类覆盖了 Object 类的 equals() hashCode() 方法,以确保两个相等的集返回相同的哈希码。若两个集大小相等且包含相同元素,则这两个集相等。按定义,集的哈希码是集中元素哈希码的总和。因此,不论集的内部顺序如何,两个相等的集会有相同的哈希码。
3.4.1. Object
(1)  boolean equals(Object obj): 对两个对象进行比较,以便确定它们是否相同
(2)  int hashCode(): 返回该对象的哈希码。相同的对象必须返回相同的哈希码
3.5 . HashSet TreeSet
“集合框架”支持 Set 接口两种普通的实现: HashSet TreeSet(TreeSet 实现 SortedSet 接口 ) 。在更多情况下,您会使用 HashSet 存储重复自由的集合。考虑到效率,添加到 HashSet 的对象需要采用恰当分配哈希码的方式来实现 hashCode() 方法。虽然大多数系统类覆盖了 Object 中缺省的 hashCode() equals() 实现,但创建您自己的要添加到 HashSet 的类时,别忘了覆盖 hashCode() equals()
当您要从集合中以有序的方式插入和抽取元素时, TreeSet 实现会有用处。为了能顺利进行,添加到 TreeSet 的元素必须是可排序的。
3.5 .1.HashSet
(1)  HashSet(): 构建一个空的哈希集
(2)  HashSet(Collection c): 构建一个哈希集,并且添加集合 c 中所有元素
(3)  HashSet(int initialCapacity): 构建一个拥有特定容量的空哈希集
(4)  HashSet(int initialCapacity, float loadFactor): 构建一个拥有特定容量和加载因子的空哈希集。 LoadFactor 0.0 1.0 之间的一个数
3.5 .2. TreeSet
(1)  TreeSet(): 构建一个空的树集
(2)  TreeSet(Collection c): 构建一个树集,并且添加集合 c 中所有元素
(3)  TreeSet(Comparator c): 构建一个树集,并且使用特定的比较器对其元素进行排序
comparator 比较器没有任何数据,它只是比较方法的存放器。这种对象有时称为函数对象。函数对象通常在“运行过程中”被定义为匿名内部类的一个实例。』
TreeSet(SortedSet s): 构建一个树集,添加有序集合 s 中所有元素,并且使用与有序集合 s 相同的比较器排序

 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值