链表通常由一连串结点组成,每一个结点包含任意的实例数据和指向上一个或下一个节点位置的指针
链表是一种线性表,但存储顺序不是线性,而是在每一个结点里存放下一个结点的指针
失去开箱性和有序性,但可存储数据量提升
单向链表 一个单链表的结点分为两个部分,第一部分保存数据,第二部分保存下个结点的地址,最后一个结点的第二部分为空
单向链表通过指针进行数据访问,每次访问都要从头开始
双向链表 结点分为三部分,分别为前驱指针(pre)、数据、后继指针(next),循环双向链表中第一个结点的 pre 和最后一个结点的 next 互相指向
栈和队列是线性表,操作受限
区别在于运算规则不同
链表和数组:
1 空间 数组空间连续;链表空间可连续可不连续
2 长度 链表长度可根据数据的放入自动变化;数组长度一旦定义不能改变
3 访问 数组地址连续,可以访问任意位置,遍历效率高; 每次访问需要指针重新开始,遍历效率低
4 操作 数组需要位移;链表需要改变指针的指向
5 使用场景 数据量大,需要频繁访问元素的用数组;需要进行删除、插入等操作的用链表
链表特点:
栈特点
队列特点
每个结点最多只有两个子节点的树称为二叉树
满二叉树结点数为 2^n -1,n为层数
前序遍历:先输出父节点,再遍历左子树和右子树
中序遍历:先遍历左子树,再输出父节点,最后遍历右子树
后序遍历:先遍历左子树和右子树,再输出父节点
层序遍历:逐层输出
含有红黑结点并且能自平衡的二叉树
性质
集合类似于容器,可以动态的把多个对象的引用放入集合中,可以存储数量不等的多个对象,还可以保存具有映射关系的数据
数组的存储特点
数组的存储弊端
集合特点:存储空间可变的存储模型,存储数据的容量可随时发生改变
集合分为 Collection (单列)和 Map(双列)两大类
Collection 作为集合层次的根接口只能存储对象,而其他集合不允许。有的集合有序,有的集合无序。JDK对Collection不提供直接实现,而是提供了更具体的子接口的实现,如 Set(不可重复)和 List(可重复)。Collection 多用于传递集合,在 Collection 下创建对象通用性最大(多态性)
Set的经典实现:HashSet TreeSet LinkedHashSet
List的经典实现:ArrayList(最通用) LinkedList
Map的经典实现:HashMap TreeMap
Collection接口:
public interface Collection<E> extends Iterable<E>
创建对象时只能创建实现类的对象
Collection,单列集合的顶层接口,表示一组对象(元素)
通过子接口的实现创建对象
boolean | add(E e) 确保此集合包含指定的元素(可选操作)。 |
---|---|
boolean | addAll(Collection<? extends E> c) 将指定集合中的所有元素添加到此集合(可选操作)。 |
void | clear() 从此集合中删除所有元素(可选操作)。 |
boolean | contains(Object o) 如果此集合包含指定的元素,则返回 true 。 |
boolean | containsAll(Collection<?> c) 如果此集合包含指定 集合 中的所有元素,则返回true。 |
boolean | equals(Object o) 将指定的对象与此集合进行比较以获得相等性。 |
int | hashCode() 返回此集合的哈希码值。 |
boolean | isEmpty() 如果此集合不包含元素,则返回 true 。 |
Iterator<E> | iterator() 返回此集合中的元素的迭代器。 |
boolean | remove(Object o) 从该集合中删除指定元素的单个实例(如果存在)(可选操作)。 |
boolean | removeAll(Collection<?> c) 删除指定集合中包含的所有此集合的元素(可选操作)。 |
boolean | retainAll(Collection<?> c) 仅保留此集合中包含在指定集合中的元素(可选操作)。 |
int | size() 返回此集合中的元素数。 |
Object[] | toArray() 返回一个包含此集合中所有元素的数组。 |
<T> T[] | toArray(T[] a) 返回包含此集合中所有元素的数组; 返回的数组的运行时类型是指定数组的运行时类型。 |
迭代集合的迭代器
boolean | hasNext() 如果迭代具有更多元素,则返回 true 。 |
---|---|
E | next() 返回迭代中的下一个元素。 |
hasNext 通过比较当前指针位置(cursor)与集合长度(size)是否相等,当两者相等时,cursor 指向集合外
经典实现:ArrayList LinkedList Vector
Object | add(int index, Object element) 将指定的元素插入此列表中的指定位置(可选操作)。 |
---|---|
Object | get(int index) 返回此列表中指定位置的元素。 |
int | indexOf(Object o) 返回此列表中指定元素的第一次出现的索引,如果此列表不包含元素,则返回-1。 |
int | lastIndexOf(Object o) 返回此列表中指定元素的最后一次出现的索引,如果此列表不包含元素,则返回-1。 |
ListIterator<E> | listIterator() 返回列表中的列表迭代器(按适当的顺序)。 |
E | set(int index, E element) 用指定的元素(可选操作)替换此列表中指定位置的元素。 |
List<E> | subList(int fromIndex, int toIndex) 返回此列表中指定的 fromIndex (含)和 toIndex 之间的视图。 |
List 集合的有序性体现在元素的存入顺序和迭代顺序是一致的
迭代器的使用:for循环更利于空间释放(只作用在方法内,方法结束就释放),但开发中普遍使用while
迭代器在迭代过程中,通过集合修改了集合的元素(多为添加或删除),造成迭代器获取园中判断预期修改值和实际修改值不一致
解决方案:集合自身进行遍历,在遍历过程中修改
void | add(E e) 将指定的元素插入列表(可选操作)。 |
---|---|
boolean | hasNext() 返回 true 如果遍历正向列表,列表迭代器有多个元素。 |
boolean | hasPrevious() 返回 true 如果遍历反向列表,列表迭代器有多个元素。 |
E | next() 返回列表中的下一个元素,并且前进光标位置。 |
int | nextIndex() 返回随后调用 next() 返回的元素的索引。 |
E | previous() 返回列表中的上一个元素,并向后移动光标位置。 |
int | previousIndex() 返回由后续调用 previous() 返回的元素的索引。 |
void | remove() 从列表中删除由 next() 或 previous() 返回的最后一个元素(可选操作)。 |
void | set(E e) 用 指定的元素替换由 next() 或 previous() 返回的最后一个元素(可选操作)。 |
在进行元素添加(或修改)时,加入(或修改)元素或后需要返回列表中的上一个元素,即返回到添加(或修改)的元素,否则添加(或修改)的元素不能在迭代过程中输出
在用 hasPrevious 逆序遍历列表前需要先正向遍历一次列表