12 集合框架
约 10775 字大约 36 分钟
2025-08-22
1. 集合框架概述
生活中的容器

数组的特点与弊端
- 面向对象语言对事物的体现都是以对象的形式,为了方便对多个对象进行操作,就要对对象进行存储
- 使用数组存储对象有一些弊端,
Java集合就像一种容器,可以动态地把多个对象的引用放入其中 - 数组在内存存储方面的特点
- 数组初始化以后,长度就确定了
- 数组中添加的元素是依次紧密排列的、有序的、可以重复的
- 数组声明的类型决定了元素初始化时的类型,不符合此类型的变量不能添加
- 数组可以存储基本数据类型的值,也可以存储引用数据类型的变量
- 数组在存储数据方面的弊端
- 数组初始化以后长度就不可变了,不便于扩展
- 数组中提供的属性和方法少,不便于添加、删除、插入、获取元素个数等操作,且效率不高
- 数组存储的数据特点单一,只能存储有序的、可重复的数据
Java集合框架体系
Java集合可分为Collection和Map两大体系
Collection接口:用于存储一个一个的数据,也称单列数据集合List子接口:用来存储有序的、可以重复的数据(主要用来替换数组),可以作为"动态"数组- 实现类:
ArrayList(主要实现类)LinkedListVector
- 实现类:
Set子接口:用来存储无序的、不可重复的数据(类似于高中讲的"集合")- 实现类:
HashSet(主要实现类)LinkedHashSetTreeSet
- 实现类:
Map接口:用于存储具有映射关系key-value的集合,也称双列数据集合- 实现类:
HashMap(主要实现类)LinkedHashMapTreeMapHashtableProperties
- 实现类:
JDKAPIjava.util- 集合框架全图

Collection接口继承树

Map接口继承树

集合的使用场景


2. Collection接口及方法
JDK不提供此接口的任何直接实现,而是提供更具体的子接口(Set和List等)去实现Collection接口是List和Set接口的父接口,该接口里定义的方法既可用于操作Set集合,也可用于操作List集合
添加
add(E obj):添加元素对象到当前集合中addAll(Collection other):添加other集合中的所有元素对象到当前集合中,即this = this∪other
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
public class CollectionAddTest {
@Test
public void addTest() {
// ArrayList 是 Collection 的子接口List的实现类之一
Collection coll = new ArrayList<>();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
System.out.println(coll);
}
@Test
public void addAllTest() {
Collection c1 = new ArrayList();
c1.add(1);
c1.add(2);
System.out.println("c1集合元素的个数:" + c1.size());
System.out.println("c1 = " + c1);
Collection c2 = new ArrayList();
c2.add(1);
c2.add(2);
System.out.println("c1集合元素的个数:" + c2.size());
System.out.println("c1 = " + c2);
Collection other = new ArrayList();
other.add(1);
other.add(2);
other.add(3);
System.out.println("other集合元素的个数:" + other.size());
System.out.println("other = " + other);
System.out.println();
c1.addAll(other);
System.out.println("c1集合元素的个数:" + c1.size());
System.out.println("c1.addAll(other) = " + c1);
c2.add(other);
System.out.println("c2集合元素的个数:" + c2.size());
System.out.println("c2.add(other) = " + c2);
}
}coll.addAll(other);coll.add(other);

判断
int size():获取当前集合中实际存储的元素个数boolean isEmpty():判断当前集合是否为空集合boolean contains(Object obj):当前集合是否存在与obj对象equals返回true的元素boolean containsAll(Collection coll):coll集合中的元素是否在当前集合中都存在boolean equals(Object obj):判断当前集合与obj是否相等
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
public class CollectionContainsTest {
@Test
public void test01() {
Collection coll = new ArrayList();
System.out.println("coll在添加元素之前,isEmpty = " + coll.isEmpty());
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll的元素个数" + coll.size());
System.out.println("coll在添加元素之后,isEmpty = " + coll.isEmpty());
}
@Test
public void test02() {
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll = " + coll);
System.out.println("coll是否包含“小李广” = " + coll.contains("小李广"));
System.out.println("coll是否包含“宋红康” = " + coll.contains("宋红康"));
Collection other = new ArrayList();
other.add("小李广");
other.add("扫地僧");
other.add("尚硅谷");
System.out.println("other = " + other);
System.out.println("coll.containsAll(other) = " + coll.containsAll(other));
}
@Test
public void test03() {
Collection c1 = new ArrayList();
c1.add(1);
c1.add(2);
//2
System.out.println("c1集合元素的个数:" + c1.size());
System.out.println("c1 = " + c1);
Collection c2 = new ArrayList();
c2.add(1);
c2.add(2);
//2
System.out.println("c2集合元素的个数:" + c2.size());
System.out.println("c2 = " + c2);
Collection other = new ArrayList();
other.add(1);
other.add(2);
other.add(3);
//3
System.out.println("other集合元素的个数:" + other.size());
System.out.println("other = " + other);
System.out.println();
c1.addAll(other);
//5
System.out.println("c1集合元素的个数:" + c1.size());
System.out.println("c1.addAll(other) = " + c1);
System.out.println("c1.contains(other) = " + c1.contains(other));
System.out.println("c1.containsAll(other) = " + c1.containsAll(other));
System.out.println();
c2.add(other);
System.out.println("c2集合元素的个数:" + c2.size());
System.out.println("c2.add(other) = " + c2);
System.out.println("c2.contains(other) = " + c2.contains(other));
System.out.println("c2.containsAll(other) = " + c2.containsAll(other));
}
}删除
void clear():清空集合元素boolean remove(Object obj):删除第一个找到的与obj对象equals返回true的元素boolean removeAll(Collection coll):从当前集合中删除所有与coll集合中相同的元素boolean retainAll(Collection coll):当前集合中仅保留两个集合的交集
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
public class CollectionRemoveTest {
@Test
public void test01() {
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll = " + coll);
coll.remove("小李广");
System.out.println("删除元素\"小李广\"之后coll = " + coll);
coll.clear();
System.out.println("coll清空之后,coll = " + coll);
}
@Test
public void test02() {
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll = " + coll);
Collection other = new ArrayList();
other.add("小李广");
other.add("扫地僧");
other.add("尚硅谷");
System.out.println("other = " + other);
coll.removeAll(other);
System.out.println("coll.removeAll(other)之后,coll = " + coll);
System.out.println("coll.removeAll(other)之后,other = " + other);
}
@Test
public void test03() {
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll = " + coll);
Collection other = new ArrayList();
other.add("小李广");
other.add("扫地僧");
other.add("尚硅谷");
System.out.println("other = " + other);
coll.retainAll(other);
System.out.println("coll.retainAll(other)之后,coll = " + coll);
System.out.println("coll.retainAll(other)之后,other = " + other);
}
}其它
Object[] toArray():返回包含当前集合中所有元素的数组hashCode():获取集合对象的哈希值iterator():返回迭代器对象,用于集合遍历
public class TestCollectionContains {
@Test
public void test01() {
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
//集合转换为数组:集合的toArray()方法
Object[] objects = coll.toArray();
System.out.println("用数组返回coll中所有元素:" + Arrays.toString(objects));
//对应的,数组转换为集合:调用Arrays的asList(Object ...objs)
Object[] arr1 = new Object[]{123,"AA","CC"};
Collection list = Arrays.asList(arr1);
System.out.println(list);
}
}3. Iterator(迭代器)接口
Iterator接口
- 在程序开发中,经常需要遍历集合中的所有元素。针对这种需求,
JDK专门提供了一个接口java.util.Iterator。Iterator接口也是Java集合中的一员,但它与Collection、Map接口有所不同Collection接口与Map接口主要用于存储元素Iterator被称为迭代器接口,Collection
Collection接口继承了java.lang.Iterable接口,该接口有一个iterator()方法,所有实现了Collection接口的集合类都有一个iterator()方法,用于返回一个实现了Iterator接口的对象。public Iterator iterator():获取集合对应的迭代器,用来遍历集合中的元素iterator()
Iterator接口的常用方法public E next():返回迭代的下一个元素public boolean hasNext():如果仍有元素可以迭代,则返回true
- 注意:
it.next()NoSuchElementException
package com.atguigu.iterator;
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
public class TestIterator {
@Test
public void test01(){
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
Iterator iterator = coll.iterator();
System.out.println(iterator.next());
System.out.println(iterator.next());
System.out.println(iterator.next());
//报NoSuchElementException异常
System.out.println(iterator.next());
}
@Test
public void test02(){
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
//获取迭代器对象
Iterator iterator = coll.iterator();
//判断是否还有元素可迭代
while(iterator.hasNext()) {
//取出下一个元素
System.out.println(iterator.next());
}
}
}迭代器的执行原理
Iterator迭代器对象在遍历集合时,,接下来通过一个图例来演示Iterator对象迭代元素的过程:

使用Iterator迭代器删除元素:java.util.Iterator迭代器中有一个方法void remove();
//回到起点
Iterator iter = coll.iterator();
while(iter.hasNext()){
Object obj = iter.next();
if(obj.equals("Tom")){
iter.remove();
}
}注意:
Iterator可以删除集合的元素,但是遍历过程中通过迭代器对象的remove方法,不是集合对象的remove方法- 如果还未调用
next()或在上一次调用next()方法之后已经调用了remove()方法,再调用remove()都会报IllegalStateException Collection已经有remove(xx)方法了,为什么Iterator迭代器还要提供删除方法呢?- 因为迭代器的
remove()可以按指定的条件进行删除
- 因为迭代器的
package com.atguigu.iterator;
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
public class TestIteratorRemove {
@Test
public void test01(){
Collection coll = new ArrayList();
coll.add(1);
coll.add(2);
coll.add(3);
coll.add(4);
coll.add(5);
coll.add(6);
Iterator iterator = coll.iterator();
while(iterator.hasNext()){
Integer element = (Integer) iterator.next();
if(element % 2 == 0){
iterator.remove();
}
}
System.out.println(coll);
}
}在JDK8.0时,Collection接口有了removeIf方法,可以根据条件删除(第18章)
package com.atguigu.collection;
import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
import java.util.function.Predicate;
public class TestCollectionRemoveIf {
@Test
public void test01(){
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
coll.add("佛地魔");
System.out.println("coll = " + coll);
coll.removeIf(new Predicate() {
@Override
public boolean test(Object o) {
String str = (String) o;
return str.contains("地");
}
});
System.out.println("删除包含\"地\"字的元素之后coll = " + coll);
}
}foreach循环
foreach循环(增强for循环)是JDK5.0定义的一个高级for循环,foreach循环的语法格式
for(元素的数据类型 局部变量 : Collection集合或数组){
//操作局部变量的输出操作
}
//这里局部变量就是一个临时变量,自己命名就可以import org.junit.Test;
import java.util.ArrayList;
import java.util.Collection;
public class TestForeach {
@Test
public void test01(){
Collection coll = new ArrayList();
coll.add("小李广");
coll.add("扫地僧");
coll.add("石破天");
//foreach循环其实就是使用Iterator迭代器来完成元素的遍历的。
for (Object o : coll) {
System.out.println(o);
}
}
@Test
public void test02(){
int[] nums = {1,2,3,4,5};
for (int num : nums) {
System.out.println(num);
}
System.out.println("-----------------");
String[] names = {"张三","李四","王五"};
for (String name : names) {
System.out.println(name);
}
}
}- 对于集合的遍历,
forIterator

Collection数组
public class ForTest {
public static void main(String[] args) {
String[] str = new String[5];
for (String myStr : str) {
// 此处是临时变量 修改对于外部的str数组元素来说是无效的
myStr = "atguigu";
System.out.println(myStr);
}
for (int i = 0; i < str.length; i++) {
System.out.println(str[i]);
}
}
}4. Collection子接口List
List接口特点
Java中数组用来存储数据有局限性,通常使用java.util.List来替代数组List集合类中元素有序、可重复,集合中的每个元素都有其对应的顺序索引- 举例:
List集合存储数据,就像银行门口的客服给每一个来办理业务的客户分配序号:第1个来的是张三,客服给他分配的是0;第2个来的是李四,客服给他分配的1;以此类推,最后一个序号应该是总人数-1
- 举例:

JDKAPI中List接口的实现类常用的有:ArrayListLinkedListVector
List接口方法
ListCollection
- 插入元素
void add(int index, Object ele):在index位置插入ele元素boolean addAll(int index, Collection eles):从index位置开始将eles中的所有元素添加进来
- 获取元素
Object get(int index):获取指定index位置的元素List subList(int fromIndex, int toIndex):返回从[fromIndex到toIndex)位置的子集合
- 获取元素索引
int indexOf(Object obj):返回obj在集合中首次出现的位置int lastIndexOf(Object obj):返回obj在当前集合中末次出现的位置
- 删除和替换元素
Object remove(int index):移除指定index位置的元素,并返回此元素Object set(int index, Object ele):设置指定index位置的元素为ele
import java.util.ArrayList;
import java.util.List;
public class TestListMethod {
public static void main(String[] args) {
// 创建List集合对象
List<String> list = new ArrayList<String>();
// 往 尾部添加 指定元素
list.add("图图");
list.add("小美");
list.add("不高兴");
System.out.println(list);
// add(int index,String s) 往指定位置添加
list.add(1,"没头脑");
System.out.println(list);
// String remove(int index) 删除指定位置元素 返回被删除元素
// 删除索引位置为2的元素
System.out.println("删除索引位置为2的元素");
System.out.println(list.remove(2));
System.out.println(list);
// String set(int index,String s)
// 在指定位置 进行 元素替代(改)
// 修改指定位置元素
list.set(0, "三毛");
System.out.println(list);
// String get(int index) 获取指定位置元素
// 跟size() 方法一起用 来 遍历的
for(int i = 0;i<list.size();i++){
System.out.println(list.get(i));
}
//还可以使用增强for
for (String string : list) {
System.out.println(string);
}
}
}注意:JavaSEListjava.util.Listjava.awt.List
List的实现类ArrayList
ArrayList是List接口的主要实现类- 本质上
ArrayList是对象引用的一个“变长”数组 Arrays.asList(…)方法返回的List集合,既不是ArrayList实例,也不是Vector实例,是一个固定长度的List集合(Listjava.util.ArrayListjava.util.ArraysArrayList)


List的实现类LinkedList
- 对于频繁的插入或删除元素的操作,建议使用
LinkedList类,效率较高,这是由底层采用链表(双向链表)结构存储数据决定的 - 特有方法
void addFirst(Object obj)void addLast(Object obj)Object getFirst()Object getLast()Object removeFirst()Object removeLast()
List的实现类Vector
Vector是一个古老的集合,JDK1.0就有了。ArrayListVector- 在各种
List中,最好把ArrayList作为默认选择;当插入、删除频繁时,用LinkedList;Vector总是比ArrayList - 特有方法
void addElement(Object obj)void insertElementAt(Object obj, int index)void setElementAt(Object obj, int index)void removeElement(Object obj)void removeAllElements()
练习
面试题:
@Test
public void testListRemove() {
List list = new ArrayList();
list.add(1);
list.add(2);
list.add(3);
updateList(list);
//[1,2]
System.out.println(list);
}
private static void updateList(List list) {
list.remove(2);
}5. Collection子接口Set
Set接口概述
Set接口是Collection的子接口,SetCollectionSet集合不允许包含相同元素,如果试把两个相同的元素加入同一个Set集合中,则添加操作失败Set集合支持的遍历方式和Collection集合一样:foreach和IteratorSet的常用实现类有:HashSet、TreeSet、LinkedHashSet
Set实现类HashSet
HashSet概述
HashSet是Set接口的主要实现类,大多数时候使用Set集合时都使用这个实现类HashSet按Hash算法来存储集合中的元素,HashSet具有以下特点HashSetnull
HashSet- 两个对象通过
hashCode()方法得到的哈希值相等,并且两个对象的equals()方法返回值为true
- 两个对象通过
SethashCode()equals(Object obj)HashSet集合中元素的无序性,不等同于随机,这里的无序性与元素的添加位置有关- :在添加每一个元素到数组中时,具体的存储位置是由元素的
hashCode()hash
- :在添加每一个元素到数组中时,具体的存储位置是由元素的
HashSet中添加元素的过程
- 第
1步:当向HashSet集合中存入一个元素时,HashSet会调用该对象的hashCode()方法得到该对象的hashCode值,然后根据hashCode值,通过某个散列函数决定该对象在HashSet底层数组中的存储位置 - 第
2步:如果要在数组中存储的位置上没有元素,则直接添加成功 - 第
3步:如果要在数组中存储的位置上有元素,则继续比较:- 如果两个元素的
hashCode()值不相等,则添加成功 - 如果两个元素的
hashCode()值相等,则会继续调用equals()方法:- 如果
equals()方法结果为false,则添加成功 - 如果
equals()方法结果为true,则添加失败
- 如果
- 如果两个元素的
23
举例:
import java.util.Objects;
public class MyDate {
private int year;
private int month;
private int day;
public MyDate(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
}
// equals重写
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
MyDate myDate = (MyDate) o;
return year == myDate.year &&
month == myDate.month &&
day == myDate.day;
}
// hashCode重写
@Override
public int hashCode() {
return Objects.hash(year, month, day);
}
@Override
public String toString() {
return "MyDate{" +
"year=" + year +
", month=" + month +
", day=" + day +
'}';
}
}import org.junit.Test;
import java.util.HashSet;
public class TestHashSet {
@Test
public void test01(){
HashSet set = new HashSet();
set.add("张三");
set.add("张三");
set.add("李四");
set.add("王五");
set.add("王五");
set.add("赵六");
//不允许重复,无序
System.out.println("set = " + set);
}
@Test
public void test02(){
HashSet set = new HashSet();
set.add(new MyDate(2021,1,1));
set.add(new MyDate(2021,1,1));
set.add(new MyDate(2022,2,4));
set.add(new MyDate(2022,2,4));
//不允许重复,无序
System.out.println("set = " + set);
}
}hashCode()
- 在程序运行时,同一个对象多次调用
hashCode()方法应该返回相同的值 - 两个对象
equals()方法比较返回true时,这两个对象的hashCode()方法的返回值也应相等 - 对象中用作
equals()方法比较的Field,都应该用来计算hashCode值
注意:equals()truehashCode()hashSet
equals()
- 重写
equals方法的时候一般需要同时复写hashCode方法,通常参与计算hashCode的对象的属性也应该参与到equals()中进行计算 - 推荐:开发中直接调用
Eclipse/IDEA里的快捷键自动重写equals()和hashCode()方法即可
Eclipse/IDEAhashCode31
首先,选择系数的时候要选择尽量大的系数。因为如果计算出来的
hash地址越大,所谓的“冲突”就越少,查找起来效率也会提高。(减少冲突)其次,
31只占用5bits,相乘造成数据溢出的概率较小。再次,
31可以 由i*31== (i<<5)-1来表示,现在很多虚拟机里面都有做相关优化。(提高算法效率)最后,
31是一个素数,素数作用就是如果我用一个数字来乘以这个素数,那么最终出来的结果只能被素数本身和被乘数还有1来整除!(减少冲突)
练习
- 练习
1:在List内去除重复数字值,要求尽量简单
public static List duplicateList(List list) {
HashSet set = new HashSet();
set.addAll(list);
return new ArrayList(set);
}
public static void main(String[] args) {
List list = new ArrayList();
list.add(new Integer(1));
list.add(new Integer(2));
list.add(new Integer(2));
list.add(new Integer(4));
list.add(new Integer(4));
List list2 = duplicateList(list);
for (Object integer : list2) {
System.out.println(integer);
}
}- 练习
2:获取随机数- 编写程序,获取
10个1至20的随机数,要求随机数不重复,并把最终的随机数输出到控制台
- 编写程序,获取
public class RandomValueTest {
public static void main(String[] args) {
// 创建集合对象
HashSet hs = new HashSet();
Random r = new Random();
while (hs.size() < 10) {
// 生成1到20的随机数
int num = r.nextInt(20) + 1;
hs.add(num);
}
// 遍历集合
for (Integer integer : hs) {
// 打印每一个元素
System.out.println(integer);
}
}
}- 练习
3:去重- 用
Scanner从键盘读取一行输入,去掉重复字符,打印不同的字符。比如:aaaabbbcccddd
- 用
public class DistinctTest {
public static void main(String[] args) {
// 创建键盘录入对象
Scanner sc = new Scanner(System.in);
System.out.println("请输入一行字符串:");
// 将键盘录入的字符串存储在line中
String line = sc.nextLine();
// 将字符串转换成字符数组
char[] arr = line.toCharArray();
// 创建HashSet集合对象
HashSet hs = new HashSet();
// 遍历字符数组
for (Object c : arr) {
// 将字符数组中的字符添加到集合中
hs.add(c);
}
// 遍历集合
for (Object ch : hs) {
System.out.print(ch);
}
}
}4:面试题
HashSet set = new HashSet();
Person p1 = new Person(1001,"AA");
Person p2 = new Person(1002,"BB");
set.add(p1);
set.add(p2);
p1.name = "CC";
set.remove(p1);
System.out.println(set);
set.add(new Person(1001,"CC"));
System.out.println(set);
set.add(new Person(1001,"AA"));
System.out.println(set);
//其中Person类中重写了hashCode()和equal()方法Set实现类LinkedHashSet
LinkedHashSet是HashSet的子类,不允许集合元素重复LinkedHashSet根据元素的hashCode值来决定元素的存储位置,但它同时使用双向链表维护元素的次序,这使得元素看起来是以添加顺序保存的LinkedHashSet插入性能略低于HashSet,但在迭代访问Set里的全部元素时有很好的性能

举例:
import org.junit.Test;
import java.util.LinkedHashSet;
public class TestLinkedHashSet {
@Test
public void test01(){
LinkedHashSet set = new LinkedHashSet();
set.add("张三");
set.add("张三");
set.add("李四");
set.add("王五");
set.add("王五");
set.add("赵六");
//不允许重复,体现添加顺序
System.out.println("set = " + set);
}
}Set实现类TreeSet
TreeSet概述
TreeSet是SortedSet接口的实现类,TreeSet可以按照添加元素的指定属性的大小顺序遍历TreeSet- 新增的方法如下
Comparator comparator()Object first()Object last()Object lower(Object e)Object higher(Object e)SortedSet subSet(fromElement, toElement)SortedSet headSet(toElement)SortedSet tailSet(fromElement)
TreeSet特点:TreeSet两种排序方法:自然排序、定制排序,TreeSet采用自然排序- :
TreeSetcompareTo(Object obj)- 如果试图把一个对象添加到
TreeSet时,该对象的类必须实现Comparable接口 - 实现
Comparable的类必须实现compareTo(Object obj)方法,两个对象通过compareTo(Object obj)方法的返回值来比较大小
- 如果试图把一个对象添加到
- :如果元素所属的类没有实现
Comparable接口,或不希望按照升序(默认情况)的方式排列元素或希望按照其它属性大小进行排序,则考虑使用定制排序。Comparator接口来实现,需要重写compare(T o1,T o2)- 利用
int compare(T o1,T o2)方法,比较o1和o2的大小:如果方法返回正整数,则表示o1大于o2;如果返回0,表示相等;返回负整数,表示o1小于o2 - 要实现定制排序需要将实现
Comparator接口的实例作为形参传递给TreeSet的构造器
- 利用
- :
- 因为只有相同类的两个实例才会比较大小,所以
TreeSet - 对于
TreeSet集合而言,它判断两个对象是否相等的唯一标准是:compareTo(Object obj)或compare(Object o1,Object o2)0
举例
举例1:自然排序
import org.junit.Test;
import java.util.Iterator;
import java.util.TreeSet;
public class TreeSetTest {
/*
* 自然排序:针对String类的对象
* */
@Test
public void test1(){
TreeSet set = new TreeSet();
set.add("MM");
set.add("CC");
set.add("AA");
set.add("DD");
set.add("ZZ");
//set.add(123); //报ClassCastException的异常
Iterator iterator = set.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
}
/*
* 自然排序:针对User类的对象
* */
@Test
public void test2(){
TreeSet set = new TreeSet();
set.add(new User("Tom",12));
set.add(new User("Rose",23));
set.add(new User("Jerry",2));
set.add(new User("Eric",18));
set.add(new User("Tommy",44));
set.add(new User("Jim",23));
set.add(new User("Maria",18));
//set.add("Tom");
Iterator iterator = set.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
System.out.println(set.contains(new User("Jack", 23))); //true
}
}其中,User类定义如下:
public class User implements Comparable{
String name;
int age;
public User() {
}
public User(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
/*
举例:按照age从小到大的顺序排列,如果age相同,则按照name从大到小的顺序排列
* */
public int compareTo(Object o) {
if(this == o){
return 0;
}
if(o instanceof User){
User user = (User)o;
int value = this.age - user.age;
if(value != 0){
return value;
}
return -this.name.compareTo(user.name);
}
throw new RuntimeException("输入的类型不匹配");
}
}举例2:定制排序
/*
* 定制排序
* */
@Test
public void test3(){
//按照User的姓名的从小到大的顺序排列
Comparator comparator = new Comparator() {
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof User && o2 instanceof User){
User u1 = (User)o1;
User u2 = (User)o2;
return u1.name.compareTo(u2.name);
}
throw new RuntimeException("输入的类型不匹配");
}
};
TreeSet set = new TreeSet(comparator);
set.add(new User("Tom",12));
set.add(new User("Rose",23));
set.add(new User("Jerry",2));
set.add(new User("Eric",18));
set.add(new User("Tommy",44));
set.add(new User("Jim",23));
set.add(new User("Maria",18));
//set.add(new User("Maria",28));
Iterator iterator = set.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
}练习
练习1:在一个List集合中存储了多个无大小顺序并且有重复的字符串,定义一个方法,让其有序(从小到大排序),并且不能去除重复元素。提示:考查ArrayList、TreeSet
public class SortTest {
public static void main(String[] args) {
ArrayList list = new ArrayList();
list.add("ccc");
list.add("ccc");
list.add("aaa");
list.add("aaa");
list.add("bbb");
list.add("ddd");
list.add("ddd");
sort(list);
System.out.println(list);
}
/*
* 对集合中的元素排序,并保留重复
*/
public static void sort(List list) {
TreeSet ts = new TreeSet(new Comparator() {
// 重写compare方法
@Override
public int compare(Object o1, Object o2) {
String s1 = (String)o1;
String s2 = (String)o2;
// 比较内容
int num = s1.compareTo(s2);
// 如果内容一样返回一个不为0的数字即可
return num == 0 ? 1 : num;
}
});
// 将list集合中的所有元素添加到ts中
ts.addAll(list);
// 清空list
list.clear();
// 将ts中排序并保留重复的结果在添加到list中
list.addAll(ts);
}
}6. Map接口
现实生活与开发中,我们常会看到这样的一类集合:用户ID与账户信息、学生姓名与考试成绩、IP地址与主机名等,这种对应关系就称作映射。Java提供了专门的集合框架用来存储这种映射关系的对象,即java.util.Map接口。
Map接口概述
Map与Collection并列存在,用于保存具有映射关系的数据:key-valueCollection集合称为单列集合,元素是孤立存在的Map集合称为双列集合,元素是成对存在的
MapkeyvalueStringMapkeyMap接口的常用实现类:HashMapLinkedHashMapTreeMap和Properties。HashMap是使用频率最高的实现类

Map中key-value的特点
这里主要以HashMap为例说明
HashMap中存储的key、value的特点如下:

Map中的key用Set来存放,不允许重复,Mapkey所对应的类须重写hashCode()equals()

key和value之间存在单向一对一关系,指定key总能找到唯一、确定的value,不同key对应的value可以重复,valueequals()keyvalueentryentry
Map接口的常用方法
- 添加、修改操作
Object put(Object key,Object value):将key-value添加到(或修改)当前map对象中void putAll(Map m):将m中的所有key-value对存放到当前map中
- 删除操作
Object remove(Object key):移除指定key的key-value对,并返回valuevoid clear():清空当前map中的所有数据
- 元素查询操作
Object get(Object key):获取指定key对应的valueboolean containsKey(Object key):是否包含指定的keyboolean containsValue(Object value):是否包含指定的valueint size():返回map中key-value对的个数boolean isEmpty():判断当前map是否为空boolean equals(Object obj):判断当前map和参数对象obj是否相等
- 元视图操作方法
Set keySet():返回所有key构成的Set集合Collection values():返回所有value构成的Collection集合Set entrySet():返回所有key-value对构成的Set集合
import java.util.HashMap;
public class TestMapMethod {
public static void main(String[] args) {
//创建 map对象
HashMap map = new HashMap();
//添加元素到集合
map.put("黄晓明", "杨颖");
map.put("李晨", "李小璐");
map.put("李晨", "范冰冰");
map.put("邓超", "孙俪");
System.out.println(map);
//删除指定的key-value
System.out.println(map.remove("黄晓明"));
System.out.println(map);
//查询指定key对应的value
System.out.println(map.get("邓超"));
System.out.println(map.get("黄晓明"));
}
}public static void main(String[] args) {
HashMap map = new HashMap();
map.put("许仙", "白娘子");
map.put("董永", "七仙女");
map.put("牛郎", "织女");
map.put("许仙", "小青");
System.out.println("所有的key:");
Set keySet = map.keySet();
for (Object key : keySet) {
System.out.println(key);
}
System.out.println("所有的value:");
Collection values = map.values();
for (Object value : values) {
System.out.println(value);
}
System.out.println("所有的映射关系:");
Set entrySet = map.entrySet();
for (Object mapping : entrySet) {
//System.out.println(entry);
Map.Entry entry = (Map.Entry) mapping;
System.out.println(entry.getKey() + "->" + entry.getValue());
}
}Map的实现类HashMap
HashMap概述
HashMap是Map接口使用频率最高的实现类HashMapnullnull- 存储数据采用的哈希表结构,底层使用
一维数组+单向链表+红黑树进行key-value数据的存储与HashSet一样,元素的存取顺序不能保证一致 HashMapkey:两个key的hashCode值相等,通过equals()方法返回trueHashMapvalue:两个value通过equals()方法返回true
练习
统计字符串中每个字符出现的次数String str = "aaaabbbcccccccccc";
public class WordCountTest {
public static void main(String[] args) {
String str = "aaaabbbcccccccccc";
// 将字符串转换成字符数组
char[] arr = str.toCharArray();
// 创建双列集合存储键和值
HashMap map = new HashMap();
// 遍历字符数组
for (char c : arr) {
// 如果不包含这个键
if (!map.containsKey(c)) {
// 就将键和值为1添加
map.put(c, 1);
} else {
// 如果包含这个键
// 就将键和值再加1添加进来
map.put(c, (int)map.get(c) + 1);
}
}
// 遍历双列集合
for (Object key : map.keySet()) {
System.out.println(key + "=" + map.get(key));
}
}
}Map的实现类LinkedHashMap
LinkedHashMap是HashMap的子类- 存储数据采用的
哈希表结构+链表结构,在HashMap存储结构的基础上,使用了一对双向链表来记录添加元素的先后顺序,可以保证遍历元素时,与添加的顺序一致 hashCode()equals()
public class TestLinkedHashMap {
public static void main(String[] args) {
LinkedHashMap map = new LinkedHashMap();
map.put("王五", 13000.0);
map.put("张三", 10000.0);
//key相同,新的value会覆盖原来的value
//因为String重写了hashCode和equals方法
map.put("张三", 12000.0);
map.put("李四", 14000.0);
//HashMap支持key和value为null值
String name = null;
Double salary = null;
map.put(name, salary);
Set entrySet = map.entrySet();
for (Object obj : entrySet) {
Map.Entry entry = (Map.Entry)obj;
System.out.println(entry);
}
}
}Map的实现类TreeMap
TreeMapkey-valuekey-valueTreeMapkey-valueTreeSet底层使用红黑树结构存储数据TreeMap的Key排序- 自然排序:
TreeMap的所有Key必须实现Comparable接口,且所有的Key应该是同一个类的对象,否则将会抛出ClasssCastException - 定制排序:创建
TreeMap时,构造器传入一个Comparator对象,该对象负责对TreeMap的所有key进行排序,此时不需要Map的Key实现Comparable接口
- 自然排序:
TreeMapkey:两个key通过compareTo()或者compare()返回0
public class TestTreeMap {
/*
* 自然排序举例
* */
@Test
public void test1(){
TreeMap map = new TreeMap();
map.put("CC",45);
map.put("MM",78);
map.put("DD",56);
map.put("GG",89);
map.put("JJ",99);
Set entrySet = map.entrySet();
for(Object entry : entrySet){
System.out.println(entry);
}
}
/*
* 定制排序
*
* */
@Test
public void test2(){
//按照User的姓名的从小到大的顺序排列
TreeMap map = new TreeMap(new Comparator() {
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof User && o2 instanceof User){
User u1 = (User)o1;
User u2 = (User)o2;
return u1.name.compareTo(u2.name);
}
throw new RuntimeException("输入的类型不匹配");
}
});
map.put(new User("Tom",12),67);
map.put(new User("Rose",23),"87");
map.put(new User("Jerry",2),88);
map.put(new User("Eric",18),45);
map.put(new User("Tommy",44),77);
map.put(new User("Jim",23),88);
map.put(new User("Maria",18),34);
Set entrySet = map.entrySet();
for(Object entry : entrySet){
System.out.println(entry);
}
}
}
class User implements Comparable{
String name;
int age;
public User(String name, int age) {
this.name = name;
this.age = age;
}
public User() {
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
/*
举例:按照age从小到大的顺序排列,如果age相同,则按照name从大到小的顺序排列
* */
@Override
public int compareTo(Object o) {
if(this == o){
return 0;
}
if(o instanceof User) {
User user = (User)o;
int value = this.age - user.age;
if(value != 0) {
return value;
}
return -this.name.compareTo(user.name);
}
throw new RuntimeException("输入的类型不匹配");
}
}Map的实现类Hashtable
Hashtable是Map接口的古老实现类,JDK1.0就提供了。不同于HashMap,HashtableHashtable实现原理和HashMap相同,功能相同。底层都使用哈希表结构(数组+单向链表),查询速度快- 与
HashMap一样,Hashtable也不能保证其中Key-Value对的顺序 Hashtable判断两个key相等、两个value相等的标准,与HashMap一致HashMapHashtablenullkeyvalue
面试题:Hashtable和HashMap的区别
HashMap:底层是一个哈希表(jdk7:数组+链表;jdk8:数组+链表+红黑树),是Hashtable:底层也是一个哈希表(数组+链表),是HashMap集合:可以存储null的键、null的值Hashtable集合:不能存储null的键、null的值Hashtable和Vector集合一样,在jdk1.2版本之后被更先进的集合HashMapArrayList取代了HashMap是Map的主要实现类,Hashtable是Map的古老实现类Hashtable的子类Properties(配置文件)依然活跃在历史舞台PropertiesIO
Map的实现类Properties
Properties类是Hashtable的子类,该对象-
keyvaluePropertieskeyvalue - 存取数据建议用
setProperty(String key,String value)getProperty(String key)方法
@Test
public void test01() {
Properties properties = System.getProperties();
//当前源文件字符编码
String fileEncoding = properties.getProperty("file.encoding");
System.out.println("fileEncoding = " + fileEncoding);
}
@Test
public void test02() {
Properties properties = new Properties();
properties.setProperty("user","songhk");
properties.setProperty("password","123456");
System.out.println(properties);
}
@Test
public void test03() throws IOException {
Properties pros = new Properties();
pros.load(new FileInputStream("jdbc.properties"));
String user = pros.getProperty("user");
System.out.println(user);
}7. Collections工具类
ArraysCollectionsSet ListMap
常用方法
Collections中提供了一系列静态方法对集合元素进行排序、查询和修改等操作,还提供了对集合对象设置不可变、对集合对象实现同步控制等方法
排序操作
reverse(List):反转List中元素的顺序shuffle(List):对List集合元素进行随机排序sort(List):根据元素的自然顺序对指定List集合元素按升序排序sort(List,Comparator):根据指定的Comparator对List集合元素进行排序swap(List,int, int):将指定list集合中的i处元素和j处元素进行交换
查找
Object max(Collection):根据元素的自然顺序,返回给定集合中的最大元素Object max(Collection,Comparator):根据Comparator指定的顺序,返回给定集合中的最大元素Object min(Collection):根据元素的自然顺序,返回给定集合中的最小元素Object min(Collection,Comparator):根据Comparator指定的顺序,返回给定集合中的最小元素int binarySearch(List list,T key):在List集合中查找某个元素的下标,但是List的元素必须是T或T的子类对象,而且int binarySearch(List list,T key,Comparator c):在List集合中查找某个元素的下标,但是List的元素必须是T或T的子类对象,而且cint frequency(Collection c,Object o):返回指定集合中指定元素的出现次数
复制、替换
void copy(List dest,List src):将src中的内容复制到dest中boolean replaceAll(List list,Object oldVal,Object newVal):List- 提供了多个
unmodifiableXxx()方法:返回指定Xxx的不可修改的视图
添加
boolean addAll(Collection c,T... elements):将所有指定元素添加到指定collection
同步
Collections类中提供了多个synchronizedXxx()方法:
举例
import org.junit.Test;
import java.text.Collator;
import java.util.*;
public class TestCollections {
@Test
public void test01(){
/*
public static <T> boolean addAll(Collection<? super T> c,T... elements)
将所有指定元素添加到指定 collection 中。Collection的集合的元素类型必须>=T类型
*/
Collection<Object> coll = new ArrayList<>();
Collections.addAll(coll, "hello","java");
Collections.addAll(coll, 1,2,3,4);
Collection<String> coll2 = new ArrayList<>();
Collections.addAll(coll2, "hello","java");
//Collections.addAll(coll2, 1,2,3,4);//String和Integer之间没有父子类关系
}
@Test
public void test02(){
/*
* public static <T extends Object & Comparable<? super T>> T max(Collection<? extends T> coll)
* 在coll集合中找出最大的元素,集合中的对象必须是T或T的子类对象,而且支持自然排序
*
* public static <T> T max(Collection<? extends T> coll,Comparator<? super T> comp)
* 在coll集合中找出最大的元素,集合中的对象必须是T或T的子类对象,按照比较器comp找出最大者
*
*/
List<Man> list = new ArrayList<>();
list.add(new Man("张三",23));
list.add(new Man("李四",24));
list.add(new Man("王五",25));
/*
* Man max = Collections.max(list);//要求Man实现Comparable接口,或者父类实现
* System.out.println(max);
*/
Man max = Collections.max(list, new Comparator<Man>() {
@Override
public int compare(Man o1, Man o2) {
return o2.getAge()-o2.getAge();
}
});
System.out.println(max);
}
@Test
public void test03(){
/*
* public static void reverse(List<?> list)
* 反转指定列表List中元素的顺序。
*/
List<String> list = new ArrayList<>();
Collections.addAll(list,"hello","java","world");
System.out.println(list);
Collections.reverse(list);
System.out.println(list);
}
@Test
public void test04(){
/* public static void shuffle(List<?> list)
* List 集合元素进行随机排序,类似洗牌,打乱顺序
*/
List<String> list = new ArrayList<>();
Collections.addAll(list,"hello","java","world");
Collections.shuffle(list);
System.out.println(list);
}
@Test
public void test05() {
/* public static <T extends Comparable<? super T>> void sort(List<T> list)
* 根据元素的自然顺序对指定 List 集合元素按升序排序
* public static <T> void sort(List<T> list,Comparator<? super T> c)
* 根据指定的 Comparator 产生的顺序对 List 集合元素进行排序
*/
List<Man> list = new ArrayList<>();
list.add(new Man("张三",23));
list.add(new Man("李四",24));
list.add(new Man("王五",25));
Collections.sort(list);
System.out.println(list);
Collections.sort(list, new Comparator<Man>() {
@Override
public int compare(Man o1, Man o2) {
return Collator.getInstance(Locale.CHINA).compare(o1.getName(),o2.getName());
}
});
System.out.println(list);
}
@Test
public void test06(){
/* public static void swap(List<?> list,int i,int j)
* 将指定 list 集合中的 i 处元素和 j 处元素进行交换
*/
List<String> list = new ArrayList<>();
Collections.addAll(list,"hello","java","world");
Collections.swap(list,0,2);
System.out.println(list);
}
@Test
public void test07(){
/* public static int frequency(Collection<?> c,Object o)
* 返回指定集合中指定元素的出现次数
*/
List<String> list = new ArrayList<>();
Collections.addAll(list,"hello","java","world","hello","hello");
int count = Collections.frequency(list, "hello");
System.out.println("count = " + count);
}
@Test
public void test08(){
/* public static <T> void copy(List<? super T> dest,List<? extends T> src)
* 将src中的内容复制到dest中
*/
List<Integer> list = new ArrayList<>();
for(int i=1; i<=5; i++){//1-5
list.add(i);
}
List<Integer> list2 = new ArrayList<>();
for(int i=11; i<=13; i++){//11-13
list2.add(i);
}
Collections.copy(list, list2);
System.out.println(list);
List<Integer> list3 = new ArrayList<>();
for(int i=11; i<=20; i++){//11-20
list3.add(i);
}
//java.lang.IndexOutOfBoundsException: Source does not fit in dest
//Collections.copy(list, list3);
//System.out.println(list);
}
@Test
public void test09(){
/*public static <T> boolean replaceAll(List<T> list,T oldVal,T newVal)
* 使用新值替换 List 对象的所有旧值
*/
List<String> list = new ArrayList<>();
Collections.addAll(list,"hello","java","world","hello","hello");
Collections.replaceAll(list, "hello","song");
System.out.println(list);
}
}