(Java)数据结构--ArrayList的底层实现

静态数组

Java内置数组的特点:

  • 数组的长度不可变

  • 数组只能存储同一类型的数据

  • 数组中每一个存储空间地址是连续且相等的

  • 数组提供角标的方式访问元素

Java内置数组的缺点:

  • 只有唯一的属性,length(没有其他的方法。)

  • 长度不可变,容量不可变(容量不够时,需要扩容,操作繁琐)

  • 地址连续且提供角标访问很快,但增删元素怎么办?(线性结构要保证元素的连续性,当增删元素的时候,可能会导致元素前移或后移,增删的时间复杂度是O(n),查改的时间复杂度是O(1))

动态数组:

        动态数组就是顺序存储结构在具体实现的核心思想,线性表就是动态数组的一种实现。

动态数组:

- 线性表
- 栈
- 双端栈
- 队列
- 循环队列
- 双端队列

线性表的定义

- 零个或多个元素的有限序列

        在储存一大波数的时候,我们通常使用的时数组(数组的出现主要是解决多数据,节约变量名。 ),但有些时候数组显得不够灵活。数组中只有一个属性。

创建List接口

package 数据结构接口;

import java.util.Comparator;

public interface List<E> extends Iterable<E>{
    public void add(E element);
    public void add(int index, E element);
    public void remove(E element);
    public E remove(int index);
    public E get(int index);
    public E set(int index, E element);
    public int size();
    public int indexOf(E element);
    public boolean contains(E element);
    public boolean isEmpty();
    public void clear();
    public void sort(Comparator<E> c);
    public List<E> subList(int formIndex, int toIndex);
}

创建ArrayList类并继承List接口


package par_01实现;

import 数据结构接口.List;

import java.util.Arrays;
import java.util.Comparator;
import java.util.Iterator;

public class ArrayList<E> implements List<E> {
    //线性表的默认容量
    private static int DEFAULT_CAPACITY = 10;
    //线性表储存元素的容器 data.length 容器的最大容量
    private E[] data;
    //线性表中元素的有效个数(表长), 也表示新来元素默认在尾部添加的角标
    private int size;

    //默认创建一个容量为DEFAULT_CAPACITY的线性表
    public ArrayList(){
        this(DEFAULT_CAPACITY);
    }

    //创建一个有用户指定容量capacity的线性表
    public ArrayList( int capacity){
        if(capacity < 0){
            throw new IllegalArgumentException("初始化的线性表的默认值必须大于等于0");
        }
        data = (E[]) new Object[capacity];
        size = 0;
    }
    //将一个一维数组转换成一个线性表
    public ArrayList(E[] arr){
        if(arr == null){
            throw new IllegalArgumentException("初始化的值不能设置为空");
        }
        data = (E[]) new Object[arr.length];
        for (int a = 0; a < arr.length; a++) {
            data[a] = arr[a];
        }
        size = arr.length;
    }

    //在线性表的末尾添加一个元素
    @Override
    public void add(E element) {
        add(size, element);
    }

    //在线性表指定的位置上添加一个元素
    /*
    插入算法的思路:
    如果插入位置不合理,抛出异常
    如果线性表满了,需要扩容
    从最后一个元素开始向前遍历到第位置,分别将它们向后移动一个位置
    将要插入得位置填入位置i处
    size加1.
     */
    @Override
    public void add(int index, E element) {

        if(index < 0 || index > size){
            throw new IndexOutOfBoundsException("下标越界异常");
        }
        //判断线性表是否已经满了
        if (size == data.length){
            resize(data.length * 2);
        }
        for (int i = size - 1; i >= index ; i--) {
            data[i + 1] = data[i];
        }
        data[index] = element;
        size++;
    }

    private void resize(int newLength) {
        //1.先创建一个新数组
        E[] newData = (E[]) new Object[newLength];
        //2.将原数组daat中的元素复制到新数组newData中
        for (int i = 0; i < size; i++) {
            newData[i] = data[i];
        }
        data = newData;
    }

    //删除指定元素

    @Override
    public void remove(E element) {
       //获取该元素的角标
        int index = indexOf(element);
        //判断元素的存在性,如果存在就根据角标删除该元素
        if(index != -1){
            remove(index);
        }
    }

    //删除指定角标的元素,并返回被删除元素的值
    /*
    删除算法的思路:
    ·如果删除位置不合适,抛出异常
    ·取出元素
    ·从删除元素位置开始遍历到最后一个元素位置,分别将它们向前移动一个位置
    ·size减一。
     */
    @Override
    public E remove(int index) {
        if(index < 0 || index >= size){
            throw new IndexOutOfBoundsException("下标越界异常");
        }
        E ret = data[index];
        for (int i = index + 1; i < size; i++) {
            data[i-1] = data[i];
        }
        size--;
        //判断是否要缩容, 不要缩容的太厉害 至少保证一个最小的值(DEFAULT_CAPACITY)
        //如果有效元素的个数是容量的1/4,且当前数组的长度大于10,则缩容
        if (size == data.length / 4 && data.length > DEFAULT_CAPACITY){
            resize(data.length / 2);
        }
        return ret;
    }

    //获取线性表中指定角标处的元素
    @Override
    public E get(int index) {
        if(index < 0 || index >= size){
            throw new IndexOutOfBoundsException("下标越界异常");
        }
        return data[index];
    }

    //将线性表中指定角标出的值改为element,并返回原值
    @Override
    public E set(int index, E element) {
        if(index < 0 || index >= size){
            throw new IndexOutOfBoundsException("下标越界异常");
        }
        E ret = data[index];
        data[index] = element;
        return ret;
    }

    @Override
    public int size() {
        return size;
    }

    //获取线性表中数组的容量
    //这并没有重写List接口的方法,List是链表不存在数组,所以这是ArrayList的私有方法
    public int getCapacity(){
        return data.length;
    }


    //在线性表中查找指定元素element的角标,并返回
    //从左到右第一次出现的位置,如果该元素不存在则返回-1
    @Override
    public int indexOf(E element) {
        for (int i = 0; i < size; i++){
            if (element.equals(data[i])) {
                return i;
            }
        }
        return -1;
    }

    //判断线性表中是否包含元素element
    @Override
    public boolean contains(E element) {
        return indexOf(element) != -1;
    }

    @Override
    public boolean isEmpty() {
        return size == 0;
    }

    @Override
    public void clear() {
        data = (E[]) new Object[DEFAULT_CAPACITY];
        size = 0;
    }

    @Override
    public void sort(Comparator<E> c) {
        if(c == null){
            throw new IllegalArgumentException("comparator can not be null");
        }
        int j = 0;
        E e = null;
        for (int i = 1; i < size; i++) {
            e = data[i];
            for (j = i; j > 0 && c.compare(data[j - 1], e) > 0 ; j--) {
                data[j -1] = data[j];
            }
            data[j] = e;
        }
    }

    @Override
    public List<E> subList(int formIndex, int toIndex) {
        if(formIndex < 0){
            throw  new IndexOutOfBoundsException("formIndex must >= 0 ");
        }
        if(formIndex > toIndex){
            throw  new IndexOutOfBoundsException("formIndex must <= toIndex");
        }
        if(toIndex >= size){
            throw new IndexOutOfBoundsException("toIndex must < size");
        }
        ArrayList<E> subList = new ArrayList<>();
        for (int i = formIndex; i <= toIndex; i++){
            //默认在末尾添加
            subList.add(data[i]);
        }
        return subList;
    }

    @Override
    public  boolean equals(Object obj){
        //判断是否是自己
        if (this == obj)
            return true;
        //判断obj是否和ArrayList是同一个类型
        if (getClass() != obj.getClass())
            return false;
        //判断哦比较是否为空
        if (obj == null)
            return false;
        //具体比较
        ArrayList other = (ArrayList) obj;
        //比较长度
        if (size != other.size)
            return false;
        //比较内容
        return Arrays.equals(data, other.data);
    }

    @Override
    public String toString(){
        StringBuilder sb = new StringBuilder(String.format("ArrayList :%d/%d[", size, data.length));
        if(isEmpty()){
            sb.append(']');
        } else {
            for (int i = 0; i < size; i++){
                sb.append(data[i]);
                if (i != size){
                    sb.append(',');
                }else {
                    sb.append(']');
                }
            }
        }
        return sb.toString();
    }



    @Override
    public Iterator<E> iterator() {
        return new ArrayListIterator();
    }
    class ArrayListIterator implements Iterator<E>{
        private int cur = 0;
        //判断之后是否有元素
        @Override
        public boolean hasNext() {
            return cur < size;
        }

        //先把元素返回然后再后移
        @Override
        public E next() {
            return data[cur++];
        }
    }

}

线性表顺序储存结构得优缺点:

  • 优点
    • 无需表示表中元素之间得逻辑关系而增减额外得储存空间
    • 可以快速地存取表中人一位置得元素
  • 缺点
    • 插入和删除造作需要移动大量元素
    • 当线性表长度变化较大得时候,难以确定存储空间得容量
    • 造成储存空间得碎片

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值