什么是集合
存储对象的容器,面向对象语言对事物的体现都是以对象的形式,所以为了方便对多个对象的操作,存储对象,集合是存储对象最常用的一种方式。
集合的出现就是为了持有对象。集合中可以存储任意类型的对象, 而且长度可变。在程序中有可能无法预先知道需要多少个对象, 那么用数组来装对象的话, 长度不好定义, 而集合解决了这样的问题。
集合和数组的区别
数组和集合类都是容器
数组长度是固定的,集合长度是可变的。数组中可以存储基本数据类型,集合只能存储对象数组中存储数据类型是单一的,集合中可以存储任意类型的对象。
集合类的特点
用于存储对象,长度是可变的,可以存储不同类型的对象。
数组的缺点
存储类型单一的数据容器,操作复杂(数组一旦声明好不可变)CRUD
集合的分类
集合做什么
1:将对象添加到集合
2:从集合中删除对象
3: 从集合中查找一个对象
4:从集合中修改一个对象就是增删改查
注意:集合和数组中存放的都是对象的引用而非对象本身
Java工程师对不同的容器进行了定义,虽然容器不同,但是还是有一些共性可以抽取最后抽取了一个顶层接口,那么就形成了一个集合框架。如何学习呢?当然是从顶层学起,顶层里边具有最共性,最基本的行为。具体的使用,就要选择具体的容器了。为什么? 因为不断向上抽取的东西有可能是不能创建对象的.抽象的可能性很大,并且子类对象的方法更多一些. 所以是看顶层,创建底层。那么集合的顶层是什么呢 叫做Collection
集合框架体系
---|Collection: 单列集合 ---|List: 有存储顺序, 可重复 ---|ArrayList: 数组实现, 查找快, 增删慢 由于是数组实现, 在增和删的时候会牵扯到数组 增容, 以及拷贝元素. 所以慢。数组是可以直接 按索引查找, 所以查找时较快 ---|LinkedList: 链表实现, 增删快, 查找慢 由于链表实现, 增加时只要让前一个元素记住自 己就可以, 删除时让前一个元素记住后一个元 素, 后一个元素记住前一个元素. 这样的增删效 率较高但查询时需要一个一个的遍历, 所以效率 较低 ---|Vector: 和ArrayList原理相同, 但线程安全, 效率略低 和ArrayList实现方式相同, 但考虑了线程安全问 题, 所以效率略低 ---|Set: 无存储顺序, 不可重复 ---|HashSet ---|TreeSet ---|LinkedHashSet ---| Map: 键值对 ---|HashMap ---|TreeMap ---|HashTable ---|LinkedHashMap
为什么出现这么多集合容器,因为每一个容器对数据的存储方式不同,这种存储方式称之为数据结构(data structure)
注意 集合和数组中存放的都是对象的引用。
什么时候该使用什么样的集合
Collection 我们需要保存若干个对象的时候使用集合。
List
如果我们需要保留存储顺序, 并且保留重复元素, 使用List.
如果查询较多, 那么使用ArrayList
如果存取较多, 那么使用LinkedList
如果需要线程安全, 那么使用Vector
Set
如果我们不需要保留存储顺序, 并且需要去掉重复元素, 使用Set.
如果我们需要将元素排序, 那么使用TreeSet
如果我们不需要排序, 使用HashSet, HashSet比
TreeSet效率高.
如果我们需要保留存储顺序, 又要过滤重复元素, 那么使用LinkedHashSet
集合类(Collection)
Collection接口有两个子接口:
List(链表|线性表)
Set(集)
特点:
Collection中描述的是集合共有的功能(CRUD)
List可存放重复元素,元素存取是有序的
Set不可以存放重复元素,元素存取是无序的
java.util.Collection ---| Collection 描述所有接口的共性 ----| List接口 可以有重复元素的集合 ----| Set 接口 不可以有重复元素的集合
学习集合对象
学习Collection中的共性方法,多个容器在不断向上抽取就出现了该体系。发现Collection接口中具有所有容器都具备的共性方法。查阅API时,就可以直接看该接口中的方法。并创建其子类对象对集合进行基本应用。当要使用集合对象中特有的方法,在查看子类具体内容。
查看api 文档Collection在在java.util 中(注意是大写Collection)
注意在现阶段遇到的 E T 之类的类型,需要暂时理解为object 因为涉及到了泛型.
3:创建集合对象,使用Collection中的List的具体实现类ArrayList
1:Collection coll=new Arraylist();
Collection接口的共性方法
增加: 1:add() 将指定对象存储到容器中 add 方法的参数类型是Object 便于接收任意对象 2:addAll() 将指定集合中的元素添加到调用该方法和集合中 删除: 3:remove() 将指定的对象从集合中删除 4:removeAll() 将指定集合中的元素删除 修改 5:clear() 清空集合中的所有元素 判断 6:isEmpty() 判断集合是否为空 7:contains() 判断集合何中是否包含指定对象 8:containsAll() 判断集合中是否包含指定集合 使用equals()判断两个对象是否相等 获取: 9:int size() 返回集合容器的大小 转成数组10: toArray() 集合转换数组
List
---| Iterable 接口 Iterator iterator() ----| Collection 接口 ------| List 接口 元素可以重复,允许在指定位置插入元素,并通过索 引来访问元素
List集合特有方法
增加 void add(int index, E element) 指定位置添加元素 boolean addAll(int index, Collection c) 指定位置添加集合 2:删除 E remove(int index) 删除指定位置元素 3:修改 E set(int index, E element) 返回的是需要替换的集合中的元素 4:查找: E get(int index) 注意: IndexOutOfBoundsException int indexOf(Object o) // 找不到返回-1 lastIndexOf(Object o) 5:求子集合 List<E> subList(int fromIndex, int toIndex) // 不包含toIndex
ArrayList
--| Iterable ----| Collection ------| List ---------| ArrayList 底层采用数组实现,默认10。每次增长 60%,((oldCapacity * 3)/2 + 1) 查询快,增删慢。 ---------| LinkedList
ArrayList:实现原理:
数组实现, 查找快, 增删慢
数组为什么是查询快?因为数组的内存空间地址是连续的.
ArrayList底层维护了一个Object[] 用于存储对象,默认数组的长度是10。可以通过 new ArrayList(20)显式的指定用于存储对象的数组的长度。
当默认的或者指定的容量不够存储对象的时候,容量自动增长为原来的容量的1.5倍。
由于ArrayList是数组实现, 在增和删的时候会牵扯到数组增容, 以及拷贝元素. 所以慢。数组是可以直接按索引查找, 所以查找时较快
可以考虑,假设向数组的0角标未知添加元素,那么原来的角标位置的元素需要整体往后移,并且数组可能还要增容,一旦增容,就需要要将老数组的内容拷贝到新数组中.所以数组的增删的效率是很低的.
LinkedList
--| Iterable ----| Collection ------| List ---------| ArrayList 底层采用数组实现,默认10。每次增长 60%,((oldCapacity * 3)/2 + 1) 查询快,增删慢。 ---------| LinkedList 底层采用链表实现,增删快,查询慢。
LinkedList:链表实现, 增删快, 查找慢
由于LinkedList:在内存中的地址不连续,需要让上一个元素记住下一个元素.所以每个元素中保存的有下一个元素的位置.虽然也有角标,但是查找的时候,需要从头往下找,显然是没有数组查找快的.但是,链表在插入新元素的时候,只需要让前一个元素记住新元素,让新元素记住下一个元素就可以了.所以插入很快.
由于链表实现, 增加时只要让前一个元素记住自己就可以, 删除时让前一个元素记住后一个元素, 后一个元素记住前一个元素. 这样的增删效率较高。
但查询时需要一个一个的遍历, 所以效率较低。
特有方法
1:方法介绍 addFirst(E e) addLast(E e) getFirst() getLast() removeFirst() removeLast() 如果集合中没有元素,获取或者删除元 素抛:NoSuchElementException 2:数据结构 1:栈 (1.6) 先进后出 push() pop() 2:队列(双端队列1.5) 先进先出 offer() poll() 3:返回逆序的迭代器对象 descendingIterator() 返回逆序的迭代器对象
ArrayList 和 LinkedList的存储查找的优缺点:
1、ArrayList 是采用动态数组来存储元素的,它允许直接用下标号来直接查找对应的元素。但是,但是插入元素要涉及数组元素移动及内存的操作。总结:查找速度快,插入操作慢。
2、LinkedList 是采用双向链表实现存储,按序号索引数据需要进行前向或后向遍历,但是插入数据时只需要记录本项的前后项即可,所以插入速度较快
问题:有一批数据要存储,要求存储这批数据不能出现重复数据,ArrayList、LinkedList都没法满足需求。解决办法:使用 set集合。
Vector
Vector: 描述的是一个线程安全的ArrayList。
ArrayList: 单线程效率高
Vector : 多线程安全的,所以效率低
特有的方法: void addElement(E obj) 在集合末尾添加元素 E elementAt( int index) 返回指定角标的元素 Enumeration elements() 返回集合中的所有元素,封装到Enumeration对象中 Enumeration 接口: boolean hasMoreElements() 测试此枚举是否包含更多的元素。 E nextElement() 如果此枚举对象至少还有一个可提供的元素,则返回此枚举的下一个元素。
迭代器
为了方便的处理集合中的元素,Java中出现了一个对象,该对象提供了一些方法专门处理集合中的元素.例如删除和获取集合中的元素.该对象就叫做迭代器(Iterator).
对 Collection 进行迭代的类,称其为迭代器。还是面向对象的思想,专业对象做专业的事情,迭代器就是专门取出集合元素的对象。但是该对象比较特殊,不能直接创建对象(通过new),该对象是以内部类的形式存在于每个集合类的内部。
如何获取迭代器?Collection接口中定义了获取集合类迭代器的方法(iterator()),所以所有的Collection体系集合都可以获取自身的迭代器。
正是由于每一个容器都有取出元素的功能。这些功能定义都一样,只不过实现的具体方式不同(因为每一个容器的数据结构不一样)所以对共性的取出功能进行了抽取,从而出现了Iterator接口。而每一个容器都在其内部对该接口进行了内部类的实现。也就是将取出方式的细节进行封装。
Iterable
Jdk1.5之后添加的新接口, Collection的父接口. 实现了Iterable的类就是可迭代的.并且支持增强for循环。该接口只有一个方法即获取迭代器的方法iterator()可以获取每个容器自身的迭代器Iterator。(Collection)集合容器都需要获取迭代器(Iterator)于是在5.0后又进行了抽取将获取容器迭代器的iterator()方法放入到了Iterable接口中。Collection接口进程了Iterable,所以Collection体系都具备获取自身迭代器的方法,只不过每个子类集合都进行了重写(因为数据结构不同)
Iterator
Iterator iterator() 返回该集合的迭代器对象
该类主要用于遍历集合对象,该类描述了遍历集合的常见方法 1:java.lang. Itreable ---| Itreable 接口 实现该接口可以使用增强for循环 ---| Collection 描述所有集合共性的接口 ---| List接口 可以有重复元素的集合 ---| Set接口 不可以有重复元素的集合
public interface Iterable<T>
Itreable 该接口仅有一个方法,用于返回集合迭代器对象。
1: Iterator<T> iterator() 返回集合的迭代器对象
Iterator接口定义的方法
Itreator 该接口是集合的迭代器接口类,定义了常见的迭代方法 1:boolean hasNext() 判断集合中是否有元素,如果有元素可以迭代,就返回true。 2: E next() 返回迭代的下一个元素,注意: 如果没有下一个元素时,调用 next元素会抛出NoSuchElementException 3: void remove() 从迭代器指向的集合中移除迭代器返回的最后一个元素(可选操 作)。
细节一:
如果迭代器的指针已经指向了集合的末尾,那么如果再调用next()会返回NoSuchElementException异常
1 import java.util.ArrayList; 2 import java.util.Iterator; 3 4 public class Demo2 { 5 public static void main(String[] args) { 6 ArrayList list = new ArrayList(); 7 // 增加:add() 将指定对象存储到容器中 8 list.add("计算机网络"); 9 list.add("现代操作系统"); 10 list.add("java编程思想"); 11 list.add("java核心技术"); 12 list.add("java语言程序设计"); 13 System.out.println(list); 14 15 Iterator it = list.iterator(); 16 while (it.hasNext()) { 17 String next = (String) it.next(); 18 System.out.println(next); 19 } 20 // 迭代器的指针已经指向了集合的末尾 21 // String next = (String) it.next(); 22 // java.util.NoSuchElementException 23 } 24 }
细节二:
如果调用remove之前没有调用next是不合法的,会抛出IllegalStateException
1 import java.util.ArrayList; 2 import java.util.Iterator; 3 4 public class Demo2 { 5 public static void main(String[] args) { 6 ArrayList list = new ArrayList(); 7 // 增加:add() 将指定对象存储到容器中 8 list.add("计算机网络"); 9 list.add("现代操作系统"); 10 list.add("java编程思想"); 11 list.add("java核心技术"); 12 list.add("java语言程序设计"); 13 System.out.println(list); 14 15 Iterator it = list.iterator(); 16 while (it.hasNext()) { 17 // 调用remove之前没有调用next是不合法的 18 // it.remove(); 19 // java.lang.IllegalStateException 20 String next = (String) it.next(); 21 System.out.println(next); 22 } 23 24 } 25 }
迭代器原理
查看ArrayList源码
1 private class Itr implements Iterator<E> { 2 3 int cursor = 0; 4 5 int lastRet = -1; 6 7 int expectedModCount = modCount; 8 9 public boolean hasNext() { 10 return cursor != size(); 11 } 12 13 public E next() { 14 checkForComodification(); 15 try { 16 E next = get(cursor); 17 lastRet = cursor++; 18 return next; 19 } catch (IndexOutOfBoundsException e) { 20 checkForComodification(); 21 throw new NoSuchElementException(); 22 } 23 } 24 25 public void remove() { 26 if (lastRet == -1) 27 throw new IllegalStateException(); 28 checkForComodification(); 29 30 try { 31 AbstractList.this.remove(lastRet); 32 if (lastRet < cursor) 33 cursor--; 34 lastRet = -1; 35 expectedModCount = modCount; 36 } catch (IndexOutOfBoundsException e) { 37 throw new ConcurrentModificationException(); 38 } 39 } 40 41 42 }
注意在对集合进行迭代过程中,不允许出现迭代器以外的对元素的操作,因为这样会产生安全隐患,java会抛出异常并发修改异常(ConcurrentModificationException),普通迭代器只支持在迭代过程中的删除动作。
注意: ConcurrentModificationException: 当一个集合在循环中即使用引用变量操作集合又使用迭代器操作集合对象, 会抛出该异常。
1 import java.util.ArrayList; 2 import java.util.Collection; 3 import java.util.Iterator; 4 public class Demo1 { 5 public static void main(String[] args) { 6 Collection coll = new ArrayList(); 7 coll.add("aaa"); 8 coll.add("bbb"); 9 coll.add("ccc"); 10 coll.add("ddd"); 11 System.out.println(coll); 12 Iterator it = coll.iterator(); 13 while (it.hasNext()) { 14 it.next(); 15 it.remove(); 16 coll.add("abc"); // 出现了迭代器以外的对元素的操作 17 } 18 System.out.println(coll); 19 } 20 }
如果是List集合,想要在迭代中操作元素可以使用List集合的特有迭代器ListIterator,该迭代器支持在迭代过程中,添加元素和修改元素。
List特有的迭代器ListIterator
public interface ListIterator extends Iterato
ListIterator<E> listIterator()
---| Iterator hasNext() next() remove() ------| ListIterator Iterator子接口 List专属的迭代器 add(E e) 将指定的元素插入列表(可选操作)。该元素直接插入到 next 返回的下一个元素的前面(如果有) void set(E o) 用指定元素替换 next 或 previous 返回的最后一个元素 hasPrevious() 逆向遍历列表,列表迭代器有多个元素,则返回 true。 previous() 返回列表中的前一个元素。
Iterator在迭代时,只能对元素进行获取(next())和删除(remove())的操作。
对于 Iterator 的子接口ListIterator 在迭代list 集合时,还可以对元素进行添加
(add(obj)),修改set(obj)的操作。
1 import java.util.ArrayList; 2 import java.util.ListIterator; 3 4 public class Demo2 { 5 public static void main(String[] args) { 6 ArrayList list = new ArrayList(); 7 // 增加:add() 将指定对象存储到容器中 8 list.add("计算机网络"); 9 list.add("现代操作系统"); 10 list.add("java编程思想"); 11 list.add("java核心技术"); 12 list.add("java语言程序设计"); 13 System.out.println(list); 14 // 获取List专属的迭代器 15 ListIterator lit = list.listIterator(); 16 17 while (lit.hasNext()) { 18 String next = (String) lit.next(); 19 System.out.println(next); 20 } 21 22 } 23 }
Set方法:用指定元素替换 next 或 previous 返回的最后一个元素
1 import java.util.ArrayList; 2 import java.util.ListIterator; 3 4 public class Demo2 { 5 public static void main(String[] args) { 6 ArrayList list = new ArrayList(); 7 // 增加:add() 将指定对象存储到容器中 8 list.add("计算机网络"); 9 list.add("现代操作系统"); 10 list.add("java编程思想"); 11 list.add("java核心技术"); 12 list.add("java语言程序设计"); 13 System.out.println(list); 14 15 ListIterator lit = list.listIterator(); 16 lit.next(); // 计算机网络 17 lit.next(); // 现代操作系统 18 System.out.println(lit.next()); // java编程思想 19 //用指定元素替换 next 或 previous 返回的最后一个元素 20 lit.set("平凡的世界");// 将java编程思想替换为平凡的世界 21 System.out.println(list); 22 23 } 24 }
Set
Set:注重独一无二的性质,该体系集合可以知道某物是否已近存在于集合中,不会存储重复的元素
用于存储无序(存入和取出的顺序不一定相同)元素,值不能重复。
对象的相等性
引用到堆上同一个对象的两个引用是相等的。如果对两个引用调用hashCode方法,会得到相同的结果,如果对象所属的类没有覆盖Object的hashCode方法的话,hashCode会返回每个对象特有的序号(java是依据对象的内存地址计算出的此序号),所以两个不同的对象的hashCode值是不可能相等的。
如果想要让两个不同的Person对象视为相等的,就必须覆盖Object继下来的hashCode方法和equals方法,因为Object hashCode方法返回的是该对象的内存地址,所以必须重写hashCode方法,才能保证两个不同的对象具有相同的hashCode,同时也需要两个不同对象比较equals方法会返回true
该集合中没有特有的方法,直接继承自Collection。
---| Itreable 接口 实现该接口可以使用增强for循环 ---| Collection 描述所有集合共性的接口 ---| List接口 可以有重复元素的集合 ---| ArrayList ---| LinkedList ---| Set接口 不可以有重复元素的集合
set集合添加元素并使用迭代器迭代元素。
1 import java.util.HashSet; 2 import java.util.Iterator; 3 import java.util.Set; 4 5 public class Demo4 { 6 public static void main(String[] args) { 7 //Set 集合存和取的顺序不一致。 8 Set hs = new HashSet(); 9 hs.add("世界军事"); 10 hs.add("兵器知识"); 11 hs.add("舰船知识"); 12 hs.add("汉和防务"); 13 System.out.println(hs); 14 // [舰船知识, 世界军事, 兵器知识, 汉和防务] 15 Iterator it = hs.iterator(); 16 while (it.hasNext()) { 17 System.out.println(it.next()); 18 } 19 } 20 }
HashSet
---| Itreable 接口 实现该接口可以使用增强for循环 ---| Collection 描述所有集合共性的接口 ---| List接口 可以有重复元素的集合 ---| ArrayList ---| LinkedList ---| Set接口 不可以有重复元素的集合 ---| HashSet 线程不安全,存取速度快。底层是以哈希表实现的。
HashSet
哈希表边存放的是哈希值。HashSet存储元素的顺序并不是按照存入时的顺序(和List显然不同) 是按照哈希值来存的所以取数据也是按照哈希值取得。
HashSet不存入重复元素的规则.使用hashcode和equals
由于Set集合是不能存入重复元素的集合。那么HashSet也是具备这一特性的。HashSet如何检查重复?HashSet会通过元素的hashcode()和equals方法进行判断元素师否重复。
当你试图把对象加入HashSet时,HashSet会使用对象的hashCode来判断对象加入的位置。同时也会与其他已经加入的对象的hashCode进行比较,如果没有相等的hashCode,HashSet就会假设对象没有重复出现。
简单一句话,如果对象的hashCode值是不同的,那么HashSet会认为对象是不可能相等的。
因此我们自定义类的时候需要重写hashCode,来确保对象具有相同的hashCode值。
如果元素(对象)的hashCode值相同,是不是就无法存入HashSet中了? 当然不是,会继续使用equals 进行比较.如果 equals为true 那么HashSet认为新加入的对象重复了,所以加入失败。如果equals 为false那么HashSet 认为新加入的对象没有重复.新元素可以存入.
总结:
元素的哈希值是通过元素的hashcode方法 来获取的, HashSet首先判断两个元素的哈希值,如果哈希值一样,接着会比较equals方法 如果 equls结果为true ,HashSet就视为同一个元素。如果equals 为false就不是同一个元素。
哈希值相同equals为false的元素是怎么存储呢,就是在同样的哈希值下顺延(可以认为哈希值相同的元素放在一个哈希桶中)。也就是哈希一样的存一列。
hashtable
图1:hashCode值不相同的情况
图2:hashCode值相同,但equals不相同的情况。
HashSet:通过hashCode值来确定元素在内存中的位置。一个hashCode位置上可以存放多个元素。
当hashcode() 值相同equals() 返回为true 时,hashset 集合认为这两个元素是相同的元素.只存储一个(重复元素无法放入)。调用原理:先判断hashcode 方法的值,如果相同才会去判断equals 如果不相同,是不会调用equals方法的。
HashSet到底是如何判断两个元素重复。
通过hashCode方法和equals方法来保证元素的唯一性,add()返回的是boolean类型
判断两个元素是否相同,先要判断元素的hashCode值是否一致,只有在该值一致的情况下,才会判断equals方法,如果存储在HashSet中的两个对象hashCode方法的值相同equals方法返回的结果是true,那么HashSet认为这两个元素是相同元素,只存储一个(重复元素无法存入)。
注意:HashSet集合在判断元素是否相同先判断hashCode方法,如果相同才会判断equals。如果不相同,是不会调用equals方法的。
HashSet 和ArrayList集合都有判断元素是否相同的方法,
boolean contains(Object o)
HashSet使用hashCode和equals方法,ArrayList使用了equals方法
练习:使用HashSet存储字符串,并尝试添加重复字符串
回顾String类的equals()、hashCode()两个方法。
1 import java.util.HashSet; 2 import java.util.Iterator; 3 import java.util.Set; 4 5 public class Demo4 { 6 public static void main(String[] args) { 7 // Set 集合存和取的顺序不一致。 8 Set hs = new HashSet(); 9 hs.add("世界军事"); 10 hs.add("兵器知识"); 11 hs.add("舰船知识"); 12 hs.add("汉和防务"); 13 14 // 返回此 set 中的元素的数量 15 System.out.println(hs.size()); // 4 16 17 // 如果此 set 尚未包含指定元素,则返回 true 18 boolean add = hs.add("世界军事"); // false 19 System.out.println(add); 20 21 // 返回此 set 中的元素的数量 22 System.out.println(hs.size());// 4 23 Iterator it = hs.iterator(); 24 while (it.hasNext()) { 25 System.out.println(it.next()); 26 } 27 } 28 }
使用HashSet存储自定义对象,并尝试添加重复对象(对象的重复的判定)
1 package cn.itcast.gz.map; 2 3 import java.util.HashSet; 4 import java.util.Iterator; 5 6 public class Demo4 { 7 public static void main(String[] args) { 8 HashSet hs = new HashSet(); 9 hs.add(new Person("jack", 20)); 10 hs.add(new Person("rose", 20)); 11 hs.add(new Person("hmm", 20)); 12 hs.add(new Person("lilei", 20)); 13 hs.add(new Person("jack", 20)); 14 15 Iterator it = hs.iterator(); 16 while (it.hasNext()) { 17 Object next = it.next(); 18 System.out.println(next); 19 } 20 } 21 } 22 23 class Person { 24 private String name; 25 private int age; 26 27 Person() { 28 29 } 30 31 public Person(String name, int age) { 32 33 this.name = name; 34 this.age = age; 35 } 36 37 public String getName() { 38 return name; 39 } 40 41 public void setName(String name) { 42 this.name = name; 43 } 44 45 public int getAge() { 46 return age; 47 } 48 49 public void setAge(int age) { 50 this.age = age; 51 } 52 53 @Override 54 public int hashCode() { 55 System.out.println("hashCode:" + this.name); 56 return this.name.hashCode() + age * 37; 57 } 58 59 @Override 60 public boolean equals(Object obj) { 61 System.out.println(this + "---equals---" + obj); 62 if (obj instanceof Person) { 63 Person p = (Person) obj; 64 return this.name.equals(p.name) && this.age == p.age; 65 } else { 66 return false; 67 } 68 } 69 70 @Override 71 public String toString() { 72 73 return "Person@name:" + this.name + " age:" + this.age; 74 } 75 76 }
TreeSet
---| Itreable 接口 实现该接口可以使用增强for循环 ---| Collection 描述所有集合共性的接口 ---| List接口 有序,可以重复,有角标的集合 ---| ArrayList ---| LinkedList ---| Set接口 无序,不可以重复的集合 ---| HashSet 线程不安全,存取速度快。底层是以hash表实现的。 ---| TreeSet 红-黑树的数据结构,默认对元素进行自然排序(String)。如果在比较的时候两个对象返回值为0,那么元素重复。
红-黑树
红黑树是一种特定类型的二叉树
红黑树算法的规则: 左小右大。
既然TreeSet可以自然排序,那么TreeSet必定是有排序规则的。
1:让存入的元素自定义比较规则。
2:给TreeSet指定排序规则。
方式一:元素自身具备比较性
元素自身具备比较性,需要元素实现Comparable接口,重写compareTo方法,也就是让元素自身具备比较性,这种方式叫做元素的自然排序也叫做默认排序。
方式二:容器具备比较性
当元素自身不具备比较性,或者自身具备的比较性不是所需要的。那么此时可以让容器自身具备。需要定义一个类实现接口Comparator,重写compare方法,并将该接口的子类实例对象作为参数传递给TreeMap集合的构造方法。
注意:当Comparable比较方式和Comparator比较方式同时存在时,以Comparator的比较方式为主;
注意:在重写compareTo或者compare方法时,必须要明确比较的主要条件相等时要比较次要条件。(假设姓名和年龄一直的人为相同的人,如果想要对人按照年龄的大小来排序,如果年龄相同的人,需要如何处理?不能直接return 0,因为可能姓名不同(年龄相同姓名不同的人是不同的人)。此时就需要进行次要条件判断(需要判断姓名),只有姓名和年龄同时相等的才可以返回0.)
通过return 0来判断唯一性。
问题:为什么使用TreeSet存入字符串,字符串默认输出是按升序排列的?因为字符串实现了一个接口,叫做Comparable 接口.字符串重写了该接口的compareTo 方法,所以String对象具备了比较性.那么同样道理,我的自定义元素(例如Person类,Book类)想要存入TreeSet集合,就需要实现该接口,也就是要让自定义对象具备比较性.
存入TreeSet集合中的元素要具备比较性.
比较性要实现Comparable接口,重写该接口的compareTo方法
TreeSet属于Set集合,该集合的元素是不能重复的,TreeSet如何保证元素的唯一性
通过compareTo或者compare方法中的来保证元素的唯一性。
添加的元素必须要实现Comparable接口。当compareTo()函数返回值为0时,说明两个对象相等,此时该对象不会添加进来。
比较器接口
----| Comparable compareTo(Object o) 元素自身具备比较性 ----| Comparator compare( Object o1, Object o2 ) 给容器传入比较器
TreeSet集合排序的两种方式:
一,让元素自身具备比较性。
也就是元素需要实现Comparable接口,覆盖compareTo 方法。
这种方式也作为元素的自然排序,也可称为默认排序。
年龄按照搜要条件,年龄相同再比姓名。
1 import java.util.TreeSet; 2 3 public class Demo4 { 4 public static void main(String[] args) { 5 TreeSet ts = new TreeSet(); 6 ts.add(new Person("aa", 20, "男")); 7 ts.add(new Person("bb", 18, "女")); 8 ts.add(new Person("cc", 17, "男")); 9 ts.add(new Person("dd", 17, "女")); 10 ts.add(new Person("dd", 15, "女")); 11 ts.add(new Person("dd", 15, "女")); 12 13 14 System.out.println(ts); 15 System.out.println(ts.size()); // 5 16 17 } 18 } 19 20 class Person implements Comparable { 21 private String name; 22 private int age; 23 private String gender; 24 25 public Person() { 26 27 } 28 29 public Person(String name, int age, String gender) { 30 31 this.name = name; 32 this.age = age; 33 this.gender = gender; 34 } 35 36 public String getName() { 37 return name; 38 } 39 40 public void setName(String name) { 41 this.name = name; 42 } 43 44 public int getAge() { 45 return age; 46 } 47 48 public void setAge(int age) { 49 this.age = age; 50 } 51 52 public String getGender() { 53 return gender; 54 } 55 56 public void setGender(String gender) { 57 this.gender = gender; 58 } 59 60 @Override 61 public int hashCode() { 62 return name.hashCode() + age * 37; 63 } 64 65 public boolean equals(Object obj) { 66 System.err.println(this + "equals :" + obj); 67 if (!(obj instanceof Person)) { 68 return false; 69 } 70 Person p = (Person) obj; 71 return this.name.equals(p.name) && this.age == p.age; 72 73 } 74 75 public String toString() { 76 return "Person [name=" + name + ", age=" + age + ", gender=" + gender 77 + "]"; 78 } 79 80 @Override 81 public int compareTo(Object obj) { 82 83 Person p = (Person) obj; 84 System.out.println(this+" compareTo:"+p); 85 if (this.age > p.age) { 86 return 1; 87 } 88 if (this.age < p.age) { 89 return -1; 90 } 91 return this.name.compareTo(p.name); 92 } 93 94 }
二,让容器自身具备比较性,自定义比较器。
需求:当元素自身不具备比较性,或者元素自身具备的比较性不是所需的。
那么这时只能让容器自身具备。
定义一个类实现Comparator 接口,覆盖compare方法。
并将该接口的子类对象作为参数传递给TreeSet集合的构造函数。
当Comparable比较方式,及Comparator比较方式同时存在,以Comparator
比较方式为主。
1 import java.util.Comparator; 2 import java.util.TreeSet; 3 4 public class Demo5 { 5 public static void main(String[] args) { 6 TreeSet ts = new TreeSet(new MyComparator()); 7 ts.add(new Book("think in java", 100)); 8 ts.add(new Book("java 核心技术", 75)); 9 ts.add(new Book("现代操作系统", 50)); 10 ts.add(new Book("java就业教程", 35)); 11 ts.add(new Book("think in java", 100)); 12 ts.add(new Book("ccc in java", 100)); 13 14 System.out.println(ts); 15 } 16 } 17 18 class MyComparator implements Comparator { 19 20 public int compare(Object o1, Object o2) { 21 Book b1 = (Book) o1; 22 Book b2 = (Book) o2; 23 System.out.println(b1+" comparator "+b2); 24 if (b1.getPrice() > b2.getPrice()) { 25 return 1; 26 } 27 if (b1.getPrice() < b2.getPrice()) { 28 return -1; 29 } 30 return b1.getName().compareTo(b2.getName()); 31 } 32 33 } 34 35 class Book { 36 private String name; 37 private double price; 38 39 public Book() { 40 41 } 42 43 public String getName() { 44 return name; 45 } 46 47 public void setName(String name) { 48 this.name = name; 49 } 50 51 public double getPrice() { 52 return price; 53 } 54 55 public void setPrice(double price) { 56 this.price = price; 57 } 58 59 public Book(String name, double price) { 60 61 this.name = name; 62 this.price = price; 63 } 64 65 @Override 66 public String toString() { 67 return "Book [name=" + name + ", price=" + price + "]"; 68 } 69 70 }
LinkedHashSet
会保存插入的顺序。
看到array,就要想到角标。
看到link,就要想到first,last。
看到hash,就要想到hashCode,equals.
看到tree,就要想到两个接口。Comparable,Comparator。
Map
如果程序中存储了几百万个学生,而且经常需要使用学号来搜索某个学生,那么这个需求有效的数据结构就是Map。Map是一种依照键(key)存储元素的容器,键(key)很像下标,在List中下标是整数。在Map中键(key)可以使任意类型的对象。Map中不能有重复的键(Key),每个键(key)都有一个对应的值(value)。一个键(key)和它对应的值构成map集合中的一个元素。
Map中的元素是两个对象,一个对象作为键,一个对象作为值。键不可以重复,但是值可以重复。
看顶层共性方法找子类特有对象.
Map与Collection在集合框架中属并列存在
Map存储的是键值对
Map存储元素使用put方法,Collection使用add方法
Map集合没有直接取出元素的方法,而是先转成Set集合,在通过迭代获取元素
Map集合中键要保证唯一性
也就是Collection是单列集合, Map 是双列集合。
总结:
Map一次存一对元素, Collection 一次存一个。Map 的键不能重复,保证唯一。
Map 一次存入一对元素,是以键值对的形式存在.键与值存在映射关系.一定要保证键的唯一性.
查看api文档:
interface Map<K,V>
K - 此映射所维护的键的类型
V - 映射值的类型
概念
将键映射到值的对象。一个映射不能包含重复的键;每个键最多只能映射到一个值。
特点
Key和Value是1对1的关系,如:门牌号 :家 老公:老婆
双列集合
Map学习体系: ---| Map 接口 将键映射到值的对象。一个映射不能包含重复的键;每个键最多只能映射到一个值。 ---| HashMap 采用哈希表实现,所以无序 ---| TreeMap 可以对健进行排序 ---|Hashtable: 底层是哈希表数据结构,线程是同步的,不可以存入null键,null值。 效率较低,被HashMap 替代。 ---|HashMap: 底层是哈希表数据结构,线程是不同步的,可以存入null键,null值。 要保证键的唯一性,需要覆盖hashCode方法,和equals方法。 ---| LinkedHashMap: 该子类基于哈希表又融入了链表。可以Map集合进行增删提高效率。 ---|TreeMap: 底层是二叉树数据结构。可以对map集合中的键进行排序。需要使用Comparable或者Comparator 进行比较排序。return 0,来判断键的唯一性。
常见方法
1、添加: 1、V put(K key, V value) (可以相同的key值,但是添加的value值会覆 盖前面的,返回值是前一个,如果没有就返回null) 2、putAll(Map<? extends K,? extends V> m) 从指定映射中将所有映射关 系复制到此映射中(可选操作)。 2、删除 1、remove() 删除关联对象,指定key对象 2、clear() 清空集合对象 3、获取 1:value get(key); 可以用于判断键是否存在的情况。当指定的键不存在的时候,返 回的是null。 3、判断: 1、boolean isEmpty() 长度为0返回true否则false 2、boolean containsKey(Object key) 判断集合中是否包含指定的key 3、boolean containsValue(Object value) 判断集合中是否包含指定的value 4、长度: Int size()
添加:
该案例使用了HashMap,建立了学生姓名和年龄之间的映射关系。并试图添加重复的键。
1 import java.util.HashMap; 2 import java.util.Map; 3 4 public class Demo1 { 5 public static void main(String[] args) { 6 // 定义一个Map的容器对象 7 Map<String, Integer > map1 = new HashMap<String, Integer >(); 8 map1.put("jack", 20); 9 map1.put("rose", 18); 10 map1.put("lucy", 17); 11 map1.put("java", 25); 12 System.out.println(map1); 13 // 添加重复的键值(值不同),会返回集合中原有(重复键)的值, System.out.println(map1.put("jack", 30)); //20 14 15 Map<String, Integer> map2 = new HashMap<String, Integer>(); 16 map2.put("张三丰", 100); 17 map2.put("虚竹", 20); 18 System.out.println("map2:" + map2); 19 // 从指定映射中将所有映射关系复制到此映射中。 20 map1.putAll(map2); 21 System.out.println("map1:" + map1); 22 // 23 } 24 } 25 // 删除: 26 // remove() 删除关联对象,指定key对象 27 // clear() 清空集合对象 28 29 Map<String, Integer> map1 = new HashMap<String, Integer>(); 30 map1.put("jack", 20); 31 map1.put("rose", 18); 32 map1.put("lucy", 17); 33 map1.put("java", 25); 34 System.out.println(map1); 35 // 指定key,返回删除的键值对映射的值。 36 System.out.println("value:" + map1.remove("java")); 37 map1.clear(); 38 System.out.println("map1:" + map1); 39 40 41 // 获取: 42 // V get(Object key) 通过指定的key对象获取value对象 43 // int size() 获取容器的大小 44 Map<String, Integer> map1 = new HashMap<String, Integer>(); 45 map1.put("jack", 20); 46 map1.put("rose", 18); 47 map1.put("lucy", 17); 48 map1.put("java", 25); 49 System.out.println(map1); 50 // V get(Object key) 通过指定的key对象获取value对象 51 // int size() 获取容器的大小 52 System.out.println("value:" + map1.get("jack")); 53 System.out.println("map.size:" + map1.size()); 54 55 // 判断: 56 // boolean isEmpty() 长度为0返回true否则false 57 // boolean containsKey(Object key) 判断集合中是否包含指定的key 58 // boolean containsValue(Object value) 59 60 Map<String, Integer> map1 = new HashMap<String, Integer>(); 61 map1.put("jack", 20); 62 map1.put("rose", 18); 63 map1.put("lucy", 17); 64 map1.put("java", 25); 65 System.out.println(map1); 66 System.out.println("isEmpty:" + map1.isEmpty()); 67 System.out.println("containskey:" + map1.containsKey("jack")); 68 System.out.println("containsvalues:" + map1.containsValue(100));
遍历Map的方式:
1、将map 集合中所有的键取出存入set集合。 Set<K> keySet() 返回所有的key对象的Set集合 再通过get方法获取键对应的值。 2、 values() ,获取所有的值. Collection<V> values()不能获取到key对象 3、 Map.Entry对象 推荐使用 重点 Set<Map.Entry<k,v>> entrySet() 将map 集合中的键值映射关系打包成一个对象 Map.Entry对象通过Map.Entry 对象的getKey, getValue获取其键和值。
第一种方式:使用keySet
将Map转成Set集合(keySet()),通过Set的迭代器取出Set集合中的每一个元素(Iterator)就是Map集合中的所有的键,再通过get方法获取键对应的值
1 import java.util.HashMap; 2 import java.util.Iterator; 3 import java.util.Map; 4 import java.util.Set; 5 6 public class Demo2 { 7 public static void main(String[] args) { 8 Map<Integer, String> map = new HashMap<Integer, String>(); 9 map.put(1, "aaaa"); 10 map.put(2, "bbbb"); 11 map.put(3, "cccc"); 12 System.out.println(map); 13 14 // 15 // 获取方法: 16 // 第一种方式: 使用keySet 17 // 需要分别获取key和value,没有面向对象的思想 18 // Set<K> keySet() 返回所有的key对象的Set集合 19 20 Set<Integer> ks = map.keySet(); 21 Iterator<Integer> it = ks.iterator(); 22 while (it.hasNext()) { 23 Integer key = it.next(); 24 String value = map.get(key); 25 System.out.println("key=" + key + " value=" + value); 26 } 27 } 28 }
第二种方式: 通过values 获取所有值,不能获取到key对象
1 public static void main(String[] args) { 2 Map<Integer, String> map = new HashMap<Integer, String>(); 3 map.put(1, "aaaa"); 4 map.put(2, "bbbb"); 5 map.put(3, "cccc"); 6 System.out.println(map); 7 // 第二种方式: 8 // 通过values 获取所有值,不能获取到key对象 9 // Collection<V> values() 10 11 Collection<String> vs = map.values(); 12 Iterator<String> it = vs.iterator(); 13 while (it.hasNext()) { 14 String value = it.next(); 15 System.out.println(" value=" + value); 16 } 17 }
第三种方式: Map.Entry
public static interface Map.Entry<K,V>
通过Map中的entrySet()方法获取存放Map.Entry<K,V>对象的Set集合。
Set<Map.Entry<K,V>> entrySet()
面向对象的思想将map集合中的键和值映射关系打包为一个对象,就是Map.Entry
,将该对象存入Set集合,Map.Entry是一个对象,那么该对象具备的getKey,getValue获得键和值。
1 public static void main(String[] args) { 2 Map<Integer, String> map = new HashMap<Integer, String>(); 3 map.put(1, "aaaa"); 4 map.put(2, "bbbb"); 5 map.put(3, "cccc"); 6 System.out.println(map); 7 // 第三种方式: Map.Entry对象 推荐使用 重点 8 // Set<Map.Entry<K,V>> entrySet() 9 10 11 // 返回的Map.Entry对象的Set集合 Map.Entry包含了key和value对象 12 Set<Map.Entry<Integer, String>> es = map.entrySet(); 13 14 Iterator<Map.Entry<Integer, String>> it = es.iterator(); 15 16 while (it.hasNext()) { 17 18 // 返回的是封装了key和value对象的Map.Entry对象 19 Map.Entry<Integer, String> en = it.next(); 20 21 // 获取Map.Entry对象中封装的key和value对象 22 Integer key = en.getKey(); 23 String value = en.getValue(); 24 25 System.out.println("key=" + key + " value=" + value); 26 } 27 }
HashMap
底层是哈希表数据结构,线程是不同步的,可以存入null键,null值。要保证键的唯一性,需要覆盖hashCode方法,和equals方法。
案例:自定义对象作为Map的键。
1 package cn.itcast.gz.map; 2 3 import java.util.HashMap; 4 import java.util.Iterator; 5 import java.util.Map.Entry; 6 import java.util.Set; 7 8 public class Demo3 { 9 public static void main(String[] args) { 10 HashMap<Person, String> hm = new HashMap<Person, String>(); 11 hm.put(new Person("jack", 20), "1001"); 12 hm.put(new Person("rose", 18), "1002"); 13 hm.put(new Person("lucy", 19), "1003"); 14 hm.put(new Person("hmm", 17), "1004"); 15 hm.put(new Person("ll", 25), "1005"); 16 System.out.println(hm); 17 System.out.println(hm.put(new Person("rose", 18), "1006")); 18 19 Set<Entry<Person, String>> entrySet = hm.entrySet(); 20 Iterator<Entry<Person, String>> it = entrySet.iterator(); 21 while (it.hasNext()) { 22 Entry<Person, String> next = it.next(); 23 Person key = next.getKey(); 24 String value = next.getValue(); 25 System.out.println(key + " = " + value); 26 } 27 } 28 } 29 30 class Person { 31 private String name; 32 private int age; 33 34 Person() { 35 36 } 37 38 public Person(String name, int age) { 39 40 this.name = name; 41 this.age = age; 42 } 43 44 public String getName() { 45 return name; 46 } 47 48 public void setName(String name) { 49 this.name = name; 50 } 51 52 public int getAge() { 53 return age; 54 } 55 56 public void setAge(int age) { 57 this.age = age; 58 } 59 60 @Override 61 public int hashCode() { 62 63 return this.name.hashCode() + age * 37; 64 } 65 66 @Override 67 public boolean equals(Object obj) { 68 if (obj instanceof Person) { 69 Person p = (Person) obj; 70 return this.name.equals(p.name) && this.age == p.age; 71 } else { 72 return false; 73 } 74 } 75 76 @Override 77 public String toString() { 78 79 return "Person@name:" + this.name + " age:" + this.age; 80 } 81 82 } 83 }
TreeMap
TreeMap的排序,TreeMap可以对集合中的键进行排序。如何实现键的排序?
方式一:元素自身具备比较性
和TreeSet一样原理,需要让存储在键位置的对象实现Comparable接口,重写compareTo方法,也就是让元素自身具备比较性,这种方式叫做元素的自然排序也叫做默认排序。
方式二:容器具备比较性
当元素自身不具备比较性,或者自身具备的比较性不是所需要的。那么此时可以让容器自身具备。需要定义一个类实现接口Comparator,重写compare方法,并将该接口的子类实例对象作为参数传递给TreeMap集合的构造方法。
注意:当Comparable比较方式和Comparator比较方式同时存在时,以Comparator的比较方式为主;
注意:在重写compareTo或者compare方法时,必须要明确比较的主要条件相等时要比较次要条件。(假设姓名和年龄一直的人为相同的人,如果想要对人按照年龄的大小来排序,如果年龄相同的人,需要如何处理?不能直接return 0,以为可能姓名不同(年龄相同姓名不同的人是不同的人)。此时就需要进行次要条件判断(需要判断姓名),只有姓名和年龄同时相等的才可以返回0.)
通过return 0来判断唯一性。
1 import java.util.TreeMap; 2 3 public class Demo4 { 4 public static void main(String[] args) { 5 TreeMap<String, Integer> tree = new TreeMap<String, Integer>(); 6 tree.put("张三", 19); 7 tree.put("李四", 20); 8 tree.put("王五", 21); 9 tree.put("赵六", 22); 10 tree.put("周七", 23); 11 tree.put("张三", 24); 12 System.out.println(tree); 13 System.out.println("张三".compareTo("李四"));//-2094 14 } 15 }
Collections与Arrays
集合框架中的工具类:特点:该工具类中的方法都是静态的。
Collections:常见方法: 1, 对list进行二分查找: 前提该集合一定要有序。 int binarySearch(list,key); //必须根据元素自然顺序对列表进行升级排序 //要求list 集合中的元素都是Comparable 的子类。 int binarySearch(list,key,Comparator); 2,对list集合进行排序。 sort(list); //对list进行排序,其实使用的事list容器中的对象的compareTo方法 sort(list,comaprator); //按照指定比较器进行排序 3,对集合取最大值或者最小值。 max(Collection) max(Collection,comparator) min(Collection) min(Collection,comparator) 4,对list集合进行反转。 reverse(list); 5,对比较方式进行强行逆转。 Comparator reverseOrder(); Comparator reverseOrder(Comparator); 6,对list集合中的元素进行位置的置换。 swap(list,x,y); 7,对list集合进行元素的替换。如果被替换的元素不存在,那么原集合不变。 replaceAll(list,old,new); 8,可以将不同步的集合变成同步的集合。 Set synchronizedSet(Set<T> s) Map synchronizedMap(Map<K,V> m) List synchronizedList(List<T> list) 9. 如果想要将集合变数组: 可以使用Collection 中的toArray 方法。注意:是Collection不是Collections工具类 传入指定的类型数组即可,该数组的长度最好为集合的size。
Arrays:用于对数组操作的工具类
1,二分查找,数组需要有序 binarySearch(int[]) binarySearch(double[]) 2,数组排序 sort(int[]) sort(char[])…… 1, 将数组变成字符串。 toString(int[]) 2, 复制数组。 copyOf(); 3, 复制部分数组。 copyOfRange(): 4, 比较两个数组是否相同。 equals(int[],int[]); 5, 将数组变成集合。 List asList(T[]); 这样可以通过集合的操作来操作数组中元素, 但是不可以使用增删方法,add,remove。因为数组长度是固定的,会出现 UnsupportOperationExcetion。 可以使用的方法:contains,indexOf。。。 如果数组中存入的基本数据类型,那么asList会将数组实体作为集合中的元素。 如果数组中的存入的引用数据类型,那么asList会将数组中的元素作为集合中 的元素。
1 import java.util.ArrayList; 2 import java.util.Collections; 3 import java.util.Arrays; 4 import java.util.List; 5 class Demo1 6 { 7 public static void main(String[] args) 8 { 9 ArrayList<Integer> list = new ArrayList<Integer>(); 10 list.add(4); 11 list.add(3); 12 list.add(1); 13 list.add(2); 14 list.add(3); 15 // 排序 16 Collections.sort(list); 17 // 折半查找的前提是排序好的元素 18 System.out.println( Collections.binarySearch( list , 8 ) ); // 找不到返回-插入点-1 19 // 反序集合输出 20 Collections.reverse( list ); 21 System.out.println( list ); 22 // 求最值 23 System.out.println( Collections.max( list ) ); // 4 24 // fill() 使用指定的元素替换指定集合中的所有元素 25 // Collections.fill( list, 5 ); 26 System.out.println( list ); 27 28 // 将数组转换为集合 29 Integer is[] = new Integer[]{6,7,8}; 30 List<Integer> list2 = Arrays.asList(is); 31 list.addAll( list2 ); 32 System.out.println( list ); 33 34 // 将List转换为数组 35 Object [] ins = list.toArray(); 36 System.out.println( Arrays.toString( ins ) ); 37 38 39 } 40 }