一.手写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实现(简单手写)以及分析它们的区别的更多相关文章

  1. [Java]ArrayList、LinkedList、Vector、Stack的比较

    一.介绍 先回顾一下List的框架图 由图中的继承关系,可以知道,ArrayList.LinkedList.Vector.Stack都是List的四个实现类. AbstractList是一个抽象类,它 ...

  2. 10分钟教你用python 30行代码搞定简单手写识别!

    欲直接下载代码文件,关注我们的公众号哦!查看历史消息即可! 手写笔记还是电子笔记好呢? 毕业季刚结束,眼瞅着2018级小萌新马上就要来了,老腊肉小编为了咱学弟学妹们的学习,绞尽脑汁准备编一套大学秘籍, ...

  3. Java ArrayList和LinkedList

    目录 集合的概念 集合体系结构 常用list集合 list集合的特点 ArrayList LinkedList 创建对象 常用方法 遍历 ArrayList和LinkedList的区别 集合的概念 ​ ...

  4. Java ArrayList,LinkedList使用

    1.ArrayList底层采用数组实现,当使用不带参数的构造方法生成ArrayList对象时,实际上回在底层生成一个长度为10的Object类型数组. 2.如果增加的元素个数超过10个,那么Array ...

  5. 简单手写一个jqurey

    1 /** 2 * @description 手写jquery 3 * @author ddxldxl 4 */ 5 class Jquery { 6 constructor(selector) { ...

  6. ArrayList和LinkedList遍历方式及性能对比分析

    ArrayList和LinkedList的几种循环遍历方式及性能对比分析 主要介绍ArrayList和LinkedList这两种list的五种循环遍历方式,各种方式的性能测试对比,根据ArrayLis ...

  7. Java ArrayList Vector LinkedList Stack Hashtable等的差别与用法(转)

    ArrayList 和Vector是采取数组体式格式存储数据,此数组元素数大于实际存储的数据以便增长和插入元素,都容许直接序号索引元素,然则插入数据要设计到数组元素移动等内存操纵,所以索引数据快插入数 ...

  8. ajax简单手写了一个猜拳游戏

    使用ajax简单写一个猜拳游戏 HTML代码 <!DOCTYPE HTML> <html lang="en-US"> <head> <me ...

  9. Java使用poi对Execl简单_写_操作

    public class WriteExecl { @Test public void writeExeclTest() throws Exception{ OutputStream os = new ...

随机推荐

  1. for循环遍历对比

    1.for(var i=0; i<10; i++):遍历数组 var arr = ['11', '22']; for(var i=0; i<arr.length;i++){ console ...

  2. SSH三大框架整合配置详解

    首先,三大框架整合,肯定是要导入相当多的jar包,这是不容置疑的!     这里就不一一列举了,直接截图吧:             (1) 基于配置文件的整合:        第一步:我们需要在we ...

  3. TCP定时器 之 坚持定时器

    坚持定时器在接收方通告接收窗口为0,阻止发送端继续发送数据时设定. 由于连接接收端的发送窗口通告不可靠(只有数据才会确认,ACK不会确认),如果一个确认丢失了,双方就有可能因为等待对方而使连接终止:接 ...

  4. java高级面试题汇总(复习)从最基础的往上复习,每天定期更新。

    每天搬一点砖,总有一天成为大牛! 看问题的时候请不要立马去翻答案,多想想. 看完答案可以问问为什么,尝试拓展!一起加油吧! 每个答案后面都有一个小彩蛋(一个以上的拓展问题),钻研让你先人一步. jav ...

  5. leetcode241 为运算表达式设计优先级

    class Solution(object): def diffWaysToCompute(self, input): """ :type input: str :rty ...

  6. IDEA创建各种不同的工程的方法

    javaWeb工程 maven创建javaSE项目 上面点击next: 项目右下角选择自动导入: maven创建javaWeb工程 项目右下角选择自动导入maven项目 上面创建成功之后发现没有jav ...

  7. LoadRunner参数化使用mysql数据源

    因为默认是没有mysql驱动的,因此需要在网上下载一个mysql驱动 1. 在网上下载一个是MYSQL数据库的ODBC驱动程序:mysql-connector-odbc-3.51.20-win32.e ...

  8. ControlTemplate in WPF —— TextBox

    <ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" x ...

  9. Java学习笔记之ArrayList基本用法

    原文地址,转载请注明出处:https://blog.csdn.net/GongchuangSu/article/details/51514389 ArrayList简介 ArrayList是一个其容量 ...

  10. VirtualBox-5.2.8-121009-Win,虚拟机指令ifconfig不显示ip解决方法