资讯专栏INFORMATION COLUMN

JAVA基础整理(四)---手写简单的arraylist来学习arraylist

liuchengxu / 993人阅读

摘要:传入的构造对进行判断,大于的情况处理一样的,等于的话还是调用了静态的那个空对象,小于抛出非法长度的异常。查找元素差别不大,就是返回的元素。

ArrayList类

通过自定义的arraylist类与jdk源码里的ArrayList的实现的对比学习:

1.所需的变量:

private Object[] elementData;
private int size;
private static final int DEFAULT_LENGTH=10;

private static final int DEFAULT_CAPACITY = 10;
transient Object[] elementData;
private int size;

(1)默认长度设置成static final 因为这个是不随具体对象改变的,是属于类的通用不变属性。

(2)为什么要把核心数组定义成 transient 数据类型,大概是因为序列化和反序列化的过程类要自己做,不要用默认的,至于原因深入后再谈。

2.构造函数

1.无参构造

public MyList(){

elementData = new Object[DEALULT_LENGTH];

}

public ArrayList() {

this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;

}
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

无参构造可以定义成static final ,没必要每次都new一个新的空对象。

2.传入length的构造

public MyList(int length){

elementData = new Object[length];

}

public ArrayList(int initialCapacity) {

if (initialCapacity > 0) {
    this.elementData = new Object[initialCapacity];
} else if (initialCapacity == 0) {
    this.elementData = EMPTY_ELEMENTDATA;
} else {
    throw new IllegalArgumentException("Illegal Capacity: "+
                                       initialCapacity);
}

}

对length进行判断,大于0的情况处理一样的,等于0的话还是调用了静态的那个空对象,小于0抛出非法长度的异常。

3.增加元素

public void add(E obj){

Object[] newArray;
if(size == elementData.length) {
    newArray = new Object[elementData.length + (elementData.length >> 1)];
    System.arraycopy(elementData, 0, newArray, 0, elementData.length);
    elementData=newArray;
}
    elementData[size++]=obj;

}

public boolean add(E e) {

ensureCapacityInternal(size + 1);  // Increments modCount!!
elementData[size++] = e;
return true;

}

add的差别比较大,MyList的简单逻辑:判断目前存的size与elementData的长度的大小,如果size等于开的长度的话,再存进去会溢出,所以需要数组扩容,扩容的基本步骤:开一个新的数组,长度是原来的1.5倍(>>1就是加上了一半,这样运算快点),将原来数组的东西拷贝到新数组的前面,将新数组指向elementData数组,把obj存进新数组里。

源码的逻辑多了几层判断,最终的扩容操作逻辑也是大同小异的:

private void grow(int minCapacity) {

// overflow-conscious code
int oldCapacity = elementData.length;
int newCapacity = oldCapacity + (oldCapacity >> 1);
if (newCapacity - minCapacity < 0)
    newCapacity = minCapacity;
if (newCapacity - MAX_ARRAY_SIZE > 0)
    newCapacity = hugeCapacity(minCapacity);
// minCapacity is usually close to size, so this is a win:
elementData = Arrays.copyOf(elementData, newCapacity);

}

4.删除元素

public void remove(int index){

for(int i=index+1;i

}

将index后的元素前移一位,size--;

public E remove(int index) {

rangeCheck(index);
modCount++;
E oldValue = elementData(index);
int numMoved = size - index - 1;
if (numMoved > 0)
    System.arraycopy(elementData, index+1, elementData, index,
                    numMoved);
elementData[--size] = null; // clear to let GC do its work

return oldValue;

}

源码里remove后返回了删除的值,并且多了一个rangecheck的函数进行index判断,因为这个功能很多地方都可以用到,相似的。

private void rangeCheck(int index) {

if (index >= size)
    throw new IndexOutOfBoundsException(outOfBoundsMsg(index));

}

移动使用的是for循环,源码使用了copy方法,做一个时间的测试:

5.修改元素

public void set(int index,E obj){

elementData[index]=obj;

}

public E set(int index, E element) {

rangeCheck(index);

E oldValue = elementData(index);
elementData[index] = element;
return oldValue;

}

也是多了一个rangecheck的判断,然后返回了旧的的value。

6.查找元素

public E get(int index){

return (E)elementData[index];

}

public E get(int index) {

rangeCheck(index);
checkForComodification();
return ArrayList.this.elementData(offset + index);

}

差别不大,就是返回elementdata的index元素。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/77678.html

相关文章

  • 一份送给Java初学者指南

    摘要:编程思想第版这本书要常读,初学者可以快速概览,中等程序员可以深入看看,老鸟还可以用之回顾的体系。以下视频整理自慕课网工程师路径相关免费课程。 我自己总结的Java学习的系统知识点以及面试问题,目前已经开源,会一直完善下去,欢迎建议和指导欢迎Star: https://github.com/Snailclimb/Java-Guide 笔者建议初学者学习Java的方式:看书+视频+实践(初...

    banana_pi 评论0 收藏0
  • 集合Collection总览

    前言 声明,本文使用的是JDK1.8 从今天开始正式去学习Java基础中最重要的东西--->集合 无论在开发中,在面试中这个知识点都是非常非常重要的,因此,我在此花费的时间也是很多,得参阅挺多的资料,下面未必就做到日更了... 当然了,如果讲得有错的地方还请大家多多包涵并不吝在评论去指正~ 一、集合(Collection)介绍 1.1为什么需要Collection Java是一门面向对象的语言,...

    FullStackDeveloper 评论0 收藏0
  • 经过了这么多场Java面试,我明白了这些道理

    摘要:的长度为什么是的幂次方多线程并发相关问题必问创建线程的种方式。什么是线程安全。尽量少通过电话面试,效果不好。通过面试官可以大概判断这家公司的情况。 最近3个月一口气面了十几家公司的Java开发岗,大大小小的面试笔试加起来快20场,收获很多。本人毕业快2年了,毕业时在学校所在的2线省会城市找了家开发公司做java的开发,前前后后做了1年半,感觉公司对技术没有啥追求,做的项目翻来覆去就是S...

    Dean 评论0 收藏0
  • java基础巩固-泛型基础知识整理

    摘要:当某个类型变量只在整个参数列表的所有参数和返回值中的一处被应用了,那么根据调用方法时该处的实际应用类型来确定。即直接根据调用方法时传递的参数类型或返回值来决定泛型参数的类型。 标签: java [TOC] 本文对泛型的基本知识进行较为全面的总结,并附上简短的代码实例,加深记忆。 泛型 将集合中的元素限定为一个特定的类型。 术语 ArrayList -- 泛型类型 ArrayLis...

    KoreyLee 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<