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. 集合实际上是用 ...
随机推荐
- hadoop上传文件失败解决办法
hadoop上传文件到web端hdfs显示hadoop could only be replicated to 0 nodes instead of 1解决办法 错误状态:在hadoop-2.7.2目 ...
- Arduino 寻找I2C地址address
参考:http://henrysbench.capnfatz.com/henrys-bench/arduino-projects-tips-and-more/arduino-quick-tip-fin ...
- Centos7安装MySQL8.0(RPM方式)
人生处处皆学问,工作也是如此!过去不止一次在Linux上安装MySQL,可以说轻车熟路,但是写篇文章总结一下,发现有很多细节值得学习! 安装包选择 为什么用rpm? 在Linux系列上安装软件一般有源 ...
- 你知道CPU结构也会影响Redis性能吗?
啦啦啦,我是卖身不卖艺的二哈,ε=(´ο`*)))唉错啦(我是开车的二哈),我又来了,铁子们一起开车呀! 今天来分析下CPU结构对Redis性能会有影响吗? 在进行Redis性能分析的时候,通常我们会 ...
- linux 内存泄露检测工具
Valgrind Memcheck 一个强大开源的程序检测工具 下载地址:http://valgrind.org/downloads/current.html Valgrind快速入门指南:http: ...
- node_modules 文件夹需要管理员权限才能删除问题
方法一:以管理员权限运行IDE ,然后在IDE里面删除该文件夹 方法二:以管理员身份运行cmd,使用命令行来删除该文件夹 找到要删除文件夹的位置,使用命令行 rmdir /s/q 文件夹位置 /s 是 ...
- 多测师讲解selenium _滚动条定位_高级讲师肖sir
from selenium import webdriverfrom time import sleepdrvier=webdriver.Chrome()drvier.get('http://www. ...
- boost之asio
asio asio是C++的准标准网络库,并且C++20标准库的网络部分将基于ASIO,需引入头文件#include <boost/asio.hpp>. ip地址 boost定义了表示ip ...
- css变量复用 全局变量-局部变量
前言 简单使用场景:同一套后台系统有多套主题的情况下,主题色作为一个最常用到的可复用的颜色,非常有必要像js的全局变量一样存在全局变量中以作复用,之前我第一个想到的是sass的变量声明,未曾想到css ...
- 修改apt,pip,npm为国内镜像源
apt 原文件备份 sudo cp /etc/apt/sources.list /etc/apt/sources.list.bak 编辑源列表文件 sudo vim /etc/apt/sources. ...