3.2 表 ADT -3.3 Java Collection API 中的表
3.2 表 ADT
处理形如 A0, A1, A2, ……, AN-1 的一般的表。我们称这个表大小为N。将大小为0的特殊表称为空表
对于除空表以外的任何表,称 Ai-1 前驱 Ai,Ai 后继 Ai-1。
表ADT上进行操作有:
printList 打印整个表
makeEmpty 清空整个表
find 返回某一项首次出现的位置
insert 从表的某个位置插入元素
remove 从表的某个位置删除元素
findKth 返回某个位置上的元素
数组实现表
对表的操作都可以通过数组来实现。虽然数组由固定容量创建,但是需要的时候可以使用双倍的容量再创建一个数组。
在这种实现下,printList花费时间为 O(N),findKth 花费时间是常数。但是插入和删除也会花费O(N)的时间,因为在 0 位置插入或删除,会将数组元素整体后移或前移。平均下来都需要移动表的一半的元素。
表的另一种实现方式:链表
链表由一系列节点组成,每个节点均含有表元素和到包含该元素后继的节点的链。链只有后继元素的信息,并没有前驱节点的信息。
和数组实现一样,printList 花费时间为 O(N),findKth(i) 花费时间 O(i)。remove 方法可以通过修改一个 next 引用来实现,insert 则需要改变两个next 引用。
实际中更为常用的是双链表。
3.3 Java Collections API 中的表
Collection接口
public interface Collection<E> extends Iterable<E> {
int size();
boolean isEmpty();
void clear();
boolean contains(Object o);
boolean add(E e);
boolean remove(E e);
Iterator<E> iterator();
}
Collection 实现了 Iterable 接口,因此可以对其使用增强 for 循环
public static <E> void enhanceFor(Collection<E> coll) {
for (E e : coll) {
//对每一个元素操作
}
}
Iterator接口
public interface Iterator<E> {
boolean hasNext();
E next();
default void remove() {
throw new UnsupportedOperationException("remove");
}
通过 iterato r方法,每个集合都可以创建并返回一个实现 Iterator 的对象
next 方法返回下一项,hasNext 方法返回是否存在下一项。编译器遇见一个用于 Iterable 对象的增强 for 循环时,它用 iterator 方法的方法来替代增强 for 循环。如下
public static <E> void iter(Collection<E> coll) {
Iterator<E> iter = coll.iterator();
while (iter.hasNext()) {
E item = iter.next();
//对每一个元素操作
}
}
remove 方法将删除由 next 最新返回的元素。Collection 的 remove 方法必须先找出要被删除的项。而 Iterator 的 remove 方法要删除的就是当前迭代器所在位置的元素。
当直接使用 Iterator 而不是通过一个增强 for 循环间接使用时,如果对正在被迭代的集合进行结构上的改变( 即对该集合使用 add 、remove 或 clear 方法), 那么迭代器就不再合法。
只有在需要立即使用一个迭代器的时候,才应该获取迭代器。如果迭代器调用了自己的 remove 方法,那么这个迭代器依然合法,且自己的 remove 方法也只应该被调用一次。
List 接口、ArrayList 类和 LinkedList 类
List 接口继承了 Collection 接口
public interface List<E> extends Collection<E> {
E get(int index);
E set(int index, E element);
void add(int index, E element);
E remove(int index);
}
listIterator 方法将产生比通常认为还要复杂的迭代器。
ArrayList 类提供了 List ADT 的一种可增长数组的实现,对 get 和 set 的调用花费常数时间。缺点是新项插入和现有项删除代价高。LinkedList 类提供的是 List ADT 的双链表实现,优点是新项的插入和删除都是常数时间的操作,这里假设变动项的位置是已知的。LinkedList 对 get 的调用代价高。
不论 ArrayList 或是 LinkedList 作为参数被传递,makeList1 的运行时间都是 O(N) ,因为每次对add调用都是在表的末端。从而均花费常数时间(忽略ArrayList扩容的时间)
public static void makeList1(List<Integer> lst, int N) {
lst.clear();
for (int i = 0; i < N; i++) {
lst.add(i);
}
}
对于 LinkedList ,makeList2 的运行时间还是 O(N),而对于 ArrayList 运行时间是 O(N2),因为在 ArrayList 中在前端添加是一个 O(N) 操作。
public static void makeList2(List<Integer> lst, int N) {
lst.clear();
for (int i = 0; i < N; i++) {
lst.add(0,i);
}
}
ListIterator 类
扩展了 List 的 Iterator 的功能。方法 previous 和 hasPrevious 能够使表从后向前遍历
public interface ListIterator<E> extends Iterator<E> {
boolean hasPrevious();
E previous();
}
add 方法将一个新的项以当前位置放入表中,set 方法改变被迭代器看到的最后一个值,对于 LinkedList 来说很方便
3.2 表 ADT -3.3 Java Collection API 中的表的更多相关文章
- Java基础学习总结(67)——Java接口API中使用数组的缺陷
如果你发现在一个接口使用有如下定义方法: public String[] getParameters(); 那么你应该认真反思.数组不仅仅老式,而且我们有合理的理由避免暴露它们.在这篇文章中,我将试图 ...
- 利用socket模拟http的混合表单上传(在一个请求中提交表单并上传多个文件)
在非常多企业级应用中,我们都没法直接通过开发语言sdk包封装的http工具来模拟http复合表单(multipart/form-data),特别是在跨语言跨平台的编程过程中.事实上实现方 ...
- Java Collection API
在 Java2中,有一套设计优良的接口和类组成了Java集合框架Collection,使程序员操作成批的数据或对象元素极为方便.这些接口和类有很多对抽象数据类型操作的API,而这是我们常用的且在数据结 ...
- Java Hour 67 Java Collection API
本文不是一个大而全的讲述Java Coleection 相关的APi, 而是笔者认为哪些是一个初学者所能够而且必须确切知道的知识点. Collection 一脉 这里有我们比较常用的List<E ...
- Java Collection集合中的iterator方法
Iterator接口的概述 /** * java.util.Iterator接口:选代器(对集合进行遍历) * 有两个常用的方法 * boolean hasNext() * 如果仍有元素可以迭代,则返 ...
- 用java查询HBase中某表的一批数据
java代码如下: package db.query; import java.io.IOException; import org.apache.hadoop.conf.Configuration; ...
- 再说表单验证,在Web Api中使用ModelState进行接口参数验证
写在前面 上篇文章中说到了表单验证的问题,然后尝试了一下用扩展方法实现链式编程,评论区大家讨论的非常激烈也推荐了一些很强大的验证插件.其中一位园友提到了说可以使用MVC的ModelState,因为之前 ...
- Java Concurrency API 中的 Lock 接口(Lock interface) 是什么?对比同步它有什么优势?
Lock 接口比同步方法和同步块提供了更具扩展性的锁操作. 他们允许更灵活的结构,可以具有完全不同的性质,并且可以支持多个相关类的 条件对象. 它的优势有: 可以使锁更公平 可以使线程在等待锁的时候响 ...
- Java设置的读书笔记和集合框架Collection API
一个.CollectionAPI 集合是一系列对象的聚集(Collection). 集合在程序设计中是一种重要的数据接口.Java中提供了有关集合的类库称为CollectionAPI. 集合实际上是用 ...
随机推荐
- 049 01 Android 零基础入门 01 Java基础语法 05 Java流程控制之循环结构 11 break语句
049 01 Android 零基础入门 01 Java基础语法 05 Java流程控制之循环结构 11 break语句 本文知识点:break语句 break语句 break语句前情回顾 1.swi ...
- C++(VS2015)模板显式特化之template语法深入理解
首先说下遇到的情况: 这里在vc++6.0上建立了一个自定义模板类,再去覆盖这个类,分别使用部分覆盖,整体覆盖 但在vs2015上去整体覆盖类会报错. 错误如下: 错误原因:个人感觉是新版本的vs更接 ...
- interp1一维数据插值在matlab中的用法
转载:https://ww2.mathworks.cn/help/matlab/ref/interp1.html?s_tid=srchtitle#btwp6lt-2_1 interp1 一维数据插值( ...
- VS2013 c++ 生成和调用DLL动态链接库(.def 方法已验证OK)
转载:https://blog.csdn.net/zhunianguo/article/details/52294339 .def 方法 创建动态库方法: 创建动态库是生成 .dll .lib 两个个 ...
- Vue:Vue-Cli 实现的交互式的项目脚手架
一.这份文档是对应 @vue/cli.老版本的 vue-cli 文档请移步https://github.com/vuejs/vue-cli/tree/v2#vue-cli-- Vue CLI 是一个基 ...
- ansible-playbook通过github拉取部署Lnmp环境
1. 配置服务器初始化 1.1) 关闭防火墙和selinux 1 [root@test-1 ~]# /bin/systemctl stop firewalld 2 [root@test-1 ~]# ...
- ansible-playbook简介
1. ansible-playbook简介 • Playbooks 与 adhoc 相比,是一种完全不同的运用 ansible 的方式,是非常之强大的. • 简单来说,playbooks 是一种简单的 ...
- 多测师讲解 _接口自动化框架设计分层思想(001)_高级讲师肖sir
第一层: 第二层:调用接口层 VOQGWBZYNBOAVZGE
- Prometheus 入门教程(一):Prometheus 快速入门
文章首发于[陈树义]公众号,点击跳转到原文:https://mp.weixin.qq.com/s/ZXlBPHGcWeYh2hjBzacc3A Prometheus 是任何一个高级工程师必须要掌握的技 ...
- 微信小程序tabbar不显示2019.04.06
app.json中pages的第一项必须在tabBar中,且这一项需要在pages的list中(与顺序无关)否则无法显示tabBar app.json中pages数组中第一项(首页),必须在tabBa ...