【Java】 ArrayList和LinkedList实现(简单手写)以及分析它们的区别
一.手写ArrayList
public class ArrayList {
private Object[] elementData; //底层数组
private int size; //数组大小
public int size(){
/*
* 返回数组大小
*/
return size;
}
public ArrayList(){
/*
* 无参构造器,通过显式调用含参构造器
*/
this();
}
public ArrayList(int initialCapacity){
/*
* 1.含参构造器
* 2.要对传入的初始量的合法性进行检测
* 3.通过新建数组实现
*/
if(initialCapacity<){
try {
throw new Exception();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
elementData=new Object[initialCapacity];
}
public boolean isEmpty(){
/*
* 判断是否为空
*/
return size==;
}
public Object get(int index){//获取指定位置的元素
/*
* 1.获取指定下标的对象
* 2.下标合法性检测
*/
rangeCheck(index);
return elementData[index];
}
public boolean add(Object obj){//在末尾添加元素
/*
* 添加对象(不指定位置)
* 注意数组扩容
*/
ensureCapacity();
elementData[size]=obj;
size++;
return true;
}
public void add(int index,Object obj){//在指定位置添加元素
/*
* 插入操作(指定位置)
* 1.下标合法性检查
* 2.数组容量检查、扩容
* 3.数组复制(原数组,开始下标,目的数组,开始下标,长度)
*/
rangeCheck(index);
ensureCapacity();
System.arraycopy(elementData, index, elementData, index+,size-index);
elementData[index]=obj;
size++;
}
public Object remove(int index){//删除指定位置元素
/*
* 1.删除指定下标对象,并返回其值
* 2.下标合法性检测
* 3.通过数组复制实现
* 4.因为前移,数组最后一位要置为空
*/
rangeCheck(index);
int arrnums=size-index-;
Object oldValue=elementData[index];
if(arrnums>){
System.arraycopy(elementData, index+, elementData,index, arrnums);
}
elementData[--size]=null;
return oldValue;
}
public boolean remove(Object obj){//删除指定元素
/*
* 1.删除指定对象
* 2.通过遍历
* 3.equals的底层运用,找到下标,调用remove(int i)
*/
for(int i=;i<size;i++){
if(get(i).equals(obj)){ //注意底层用的是equals不是“==”
remove(i);
}
break;
}
return true;
}
public Object set(int index,Object obj){//修改指定位置的元素
/*
* 1.将指定下标的对象改变
* 2.下标合法性检查
* 3.直接通过数组的赋值来实现改变
* 4.返回旧值
*/
rangeCheck(index);
Object oldValue=elementData[index];
elementData[index]=obj;
return oldValue;
}
private void rangeCheck(int index){
/*
* 对下标的检查
*/
if(index<||index>=size){
try {
throw new Exception();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
private void ensureCapacity(){
/*
* 1.对容器容量的检查
* 2.数组扩容,通过数组复制来实现(量和值两者都要保障)
*/
if(size==elementData.length){
Object[] newArray=new Object[size*+];
System.arraycopy(elementData, , newArray, , elementData.length);
elementData=newArray;
}
}
public int indexOf(Object obj) {//查询元素第一次出现的位置
//ArrayList中的元素可以为null,如果为null返回null的下标
if (obj == null) {
for (int i = ; i < size; i++)
if (elementData[i]==null)
return i;
} else {
for (int i = ; i < size; i++)
if (obj.equals(elementData[i]))
return i;
}
//如果没有找到对应的元素返回-1。
return -;
}
public int lastIndexOf(Object obj) {//查询元素最后一次出现的位置
if (obj == null) {
//如果o为null从后往前找到第一个为null的下标
for (int i = size-; i >= ; i--)
if (elementData[i]==null)
return i;
} else {
//从后往前找到第一个值为o的下标
for (int i = size-; i >= ; i--)
if (obj.equals(elementData[i]))
return i;
}
return -;
}
}
二.手写LinkedList
package com.whzc.ywb.study.section03.linkedList;
/**
* 自己实现链表
* @author ywb
*
* @param <E>
*/
public class LinkedList<E> { private class Node{
public E e;//元素
public Node next;//指针
public Node(E e,Node next) {//传入元素和指针
this.e = e;
this.next = next;
}
public Node(){//不传入元素和指针
this(null,null);//this是传入两个参数的构造器
}
public Node(E e){
this(e,null);
} @Override
public String toString() {
return "Node [e=" + e + "]";
}
} private Node dummyHead;
private int size;
public LinkedList(){
dummyHead = new Node(null,null);
size = ;
}
public int getSize(){
return size;
}
public boolean isEmpty(){
return size == ;
}
public void addFirst(E e){//在链表头添加元素
/*Node node = new Node(e);
node.next = head;
head = node;*/ //用下面一行代码代替
//head = new Node(e,head);//括号内的head是之前的链表的头结点,左边的head是现在的头结点
add(,e);
}
public void addLast(E e){//在链表的尾部添加元素
add(size,e);
}
public void add(int index,E e){//在链表中间添加元素
if(index < || index > size){
throw new IllegalArgumentException("索引越界异常");
}
Node prev = dummyHead;//定义一个指针指向头结点
for(int i = ; i < index ; i++){
prev = prev.next;//将这个指针移动到要插入的位置的前一个元素
}
/*Node node = new Node(e);
node.next = prev.next;
prev.next = node;*/ //注意这两行代码的顺序。用下面一行代码实现
prev.next = new Node(e,prev.next);
size ++;
}
public E get(int index){
if(index < || index > size){
throw new IllegalArgumentException("索引越界异常");
}
Node cur = dummyHead.next;
for(int i = ; i < index ; i++){
cur = cur.next;
}
return cur.e;
}
public E getFirst(){
return get();
}
public E getLast(){
return get(size-);
}
public void update(int index,E e){//修改某个元素
if(index < || index > size){
throw new IllegalArgumentException("索引越界异常");
}
Node cur = dummyHead.next;
for(int i = ; i < index ; i++){
cur = cur.next;
}
cur.e = e;
}
public boolean contains(E e){//查询链表中是否存在某个元素
Node node = dummyHead.next;
while (node != null){
if(node.e.equals(e)){
return true;
}
node = node.next;
}
return false;
}
public void delete(int index){//删除元素
if(index < || index > size){
throw new IllegalArgumentException("索引越界异常");
}
Node prev = dummyHead;
for(int i = ; i < index ; i++){
prev = prev.next;
}
/*prev.next = prev.next.next;//注意!!! 这是错误的
prev.next.next = null;*/
Node cur = prev.next;
prev.next = cur.next;
cur.next = null;
size--;
}
public void deleteFirst(){
delete();
}
public void deleteLast(){
delete(size-);
}
public void deleteElement(E e){ Node prev = dummyHead;
while(prev.next != null){
if(prev.next.e.equals(e))
break;
prev = prev.next;
} if(prev.next != null){
Node delNode = prev.next;
prev.next = delNode.next;
delNode.next = null;
size --;
}
}
}
三.分析ArrayList和LinkedList的区别
从底层上分析
ArrayList的底层是由数组实现的,而LinkedList的底层是由链表实现的。
ArrayList的空间浪费主要体现在在list列表的结尾预留一定的容量空间,而LinkedList的空间花费则体现在它的每一个元素都需要消耗相当的空间
从效率上分析
1.当随机访问List时(get和set操作),ArrayList比LinkedList的效率更高,因为LinkedList是线性的数据存储方式,所以需要移动指针从前往后依次查找。
2.当对数据进行增加和删除的操作时(add和remove操作),LinkedList比ArrayList的效率更高,因为ArrayList是数组,所以在其中进行增删操作时,会对操作点之后所有数据的下标索引造成影响,需要进行数据的移动。
3.从利用效率来看,ArrayList自由性较低,因为它需要手动的设置固定大小的容量,但是它的使用比较方便,只需要创建,然后添加数据,通过调用下标进行使用;而LinkedList自由性较高,能够动态的随数据量的变化而变化,但是它不便于使用。
4.ArrayList主要控件开销在于需要在lList列表预留一定空间;而LinkList主要控件开销在于需要存储结点信息以及结点指针信息。对ArrayList和LinkedList而言,在列表末尾增加一个元素所花的开销都是固定的。对 ArrayList而言,主要是在内部数组中增加一项,指向所添加的元素,偶尔可能会导致对数组重新进行分配;而对LinkedList而言,这个开销是 统一的,分配一个内部Entry对象。
5.LinkedList集合不支持 高效的随机随机访问(RandomAccess),因为可能产生二次项的行为。
【Java】 ArrayList和LinkedList实现(简单手写)以及分析它们的区别的更多相关文章
- [Java]ArrayList、LinkedList、Vector、Stack的比较
一.介绍 先回顾一下List的框架图 由图中的继承关系,可以知道,ArrayList.LinkedList.Vector.Stack都是List的四个实现类. AbstractList是一个抽象类,它 ...
- 10分钟教你用python 30行代码搞定简单手写识别!
欲直接下载代码文件,关注我们的公众号哦!查看历史消息即可! 手写笔记还是电子笔记好呢? 毕业季刚结束,眼瞅着2018级小萌新马上就要来了,老腊肉小编为了咱学弟学妹们的学习,绞尽脑汁准备编一套大学秘籍, ...
- Java ArrayList和LinkedList
目录 集合的概念 集合体系结构 常用list集合 list集合的特点 ArrayList LinkedList 创建对象 常用方法 遍历 ArrayList和LinkedList的区别 集合的概念 ...
- Java ArrayList,LinkedList使用
1.ArrayList底层采用数组实现,当使用不带参数的构造方法生成ArrayList对象时,实际上回在底层生成一个长度为10的Object类型数组. 2.如果增加的元素个数超过10个,那么Array ...
- 简单手写一个jqurey
1 /** 2 * @description 手写jquery 3 * @author ddxldxl 4 */ 5 class Jquery { 6 constructor(selector) { ...
- ArrayList和LinkedList遍历方式及性能对比分析
ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayLis ...
- Java ArrayList Vector LinkedList Stack Hashtable等的差别与用法(转)
ArrayList 和Vector是采取数组体式格式存储数据,此数组元素数大于实际存储的数据以便增长和插入元素,都容许直接序号索引元素,然则插入数据要设计到数组元素移动等内存操纵,所以索引数据快插入数 ...
- ajax简单手写了一个猜拳游戏
使用ajax简单写一个猜拳游戏 HTML代码 <!DOCTYPE HTML> <html lang="en-US"> <head> <me ...
- Java使用poi对Execl简单_写_操作
public class WriteExecl { @Test public void writeExeclTest() throws Exception{ OutputStream os = new ...
随机推荐
- python3笔记十九:os和ospath模块
一:学习内容 os模块 ospath模块 获取指定目录下所有文件和目录 二:os模块 包含了普遍的操作系统功能,需要导入该模块:import os 当前所在位置目录结构为: 目录操作 1.获取当前目录 ...
- 网络1911、1912 C语言第2次作业--循环结构 批改总结
一.评分规则 伪代码务必是文字+代码描述,直接反应代码,每题扣1分 提交列表没内容,或者太简单,每题得分0分.注意选择提交列表长的题目介绍. 代码格式不规范,继续扣分. 代码互评,内容简单,0分. 原 ...
- cucumber+selenium
工程结构 pom <?xml version="1.0" encoding="UTF-8"?> <project xmlns="ht ...
- koa2中间键原理
一.koa2 const http = require('http'); const compose = require('./compose'); class Koa { constructor() ...
- leetcode241 为运算表达式设计优先级
class Solution(object): def diffWaysToCompute(self, input): """ :type input: str :rty ...
- 十一:jinja2模板传参
从后台传参到模板,模板再渲染到前端 传参的时候,可以在html后面加上关键字传参,在模板里面用{{ 参数 }}使用即可,可以传多个参数 也可以使用**传参,取值的时候就直接取内容
- lnmp宝塔面板问题
使用宝塔面板后,无法安装zabbix客户端的依赖包,总是提示mariadb冲突,其实mariadb早就卸载完了,所以要安装zabbix客户端就不好使用宝塔面板
- python学习之生成器
4.6 生成器Generrator 生成器本质就是迭代器.python社区生成器与迭代器是一种. 生成器与迭代器的唯一区别:生成器是我们自己用python代码构建的 4.6.1生成器初识 py ...
- [Vuejs] 在vue各个组件中应用全局scss变量
需要安装一个插件:sass-resources-loader 1.执行安装命令: npm i sass-resources-loader --save-dev 2.修改vue-cli环境下build文 ...
- Buffer对象与JSON对象相互转换
> buffer=new Buffer('换汤不换药');<Buffer e6 88 91 e7 88 b1 e4 bd a0 ef bc 8c e7 89 a9 e7 90 86> ...