模拟实现顺序表 —— ArrayList

ArrayLiist的使用

java 复制代码
import java.util.ArrayList;
import java.util.List;

// 顺序表的使用
public class Demo {
    public static void main(String[] args) {
        // 初始化顺序表
        List<Integer> list = new ArrayList<>();

        // 添加数据
        list.add(1);
        list.add(1);
        list.add(1);
        list.add(1,2);
        list.add(3);
        list.add(4);
        list.add(4);
        list.add(5);
        list.add(6);
        System.out.println(list);

        // 删除数据
        list.remove(1);
        list.remove(new Integer(1));
        System.out.println(list);

        // 更新数据
        list.set(0, 111);
        System.out.println(list);

        // 获取元素在顺序表中第一次出现的位置
        System.out.println(list.indexOf(4));
        System.out.println(list.indexOf(1));

        // 获取元素在顺序表中最后一次出现的位置
        System.out.println(list.lastIndexOf(4));
        System.out.println(list.lastIndexOf(5));

        // 获取顺序表的长度
        System.out.println(list.size());

        // 判断顺序表是否为空
        System.out.println(list.isEmpty());

        // 判断某个元素是否在顺序表中存在
        System.out.println(list.contains(111));
        System.out.println(list.contains(112));

        // 获取给定下标的元素值
        System.out.println(list.get(1));
        System.out.println(list.get(3));
    }
}

以下代码是针对上述 ArrayList 类的模拟实现

MyArrayList

java 复制代码
import java.util.Arrays;

// 模拟实现顺序表
public class MyArrayList {
    // 底层数组默认大小
    private static final int DEFAULT_CAPACITY = 10;
    private int[] elem;
    // 累计数组中的元素个数
    private int usedSize;

    // 构造器
    public MyArrayList() {
        this(DEFAULT_CAPACITY);
    }

    public MyArrayList(int capacity) {
        this.elem = new int[capacity];
    }

    // 添加数据
    public void add(int data) {
        // 判满
        if (usedSize == elem.length) {
            this.elem = Arrays.copyOf(this.elem, this.elem.length * 2);
        }
        // 添加元素到数组后面
        this.elem[usedSize++] = data;
    }

    // 根据下标添加数据
    public void add(int index, int data) {
        // 判断下标是否越界
        if (index < 0 || index > usedSize) {
            throw new IndexOfArrayException("下标异常: " + index);
        }

        // 判满
        if (usedSize == elem.length) {
            this.elem = Arrays.copyOf(this.elem, this.elem.length * 2);
        }

        // 把 index 处及其后面的下标往后移
        for (int i = usedSize; i > index; i--) {
            this.elem[i] = this.elem[i-1];
        }
        // 把要插入的元素放到 index 处
        this.elem[index] = data;
        usedSize++;
    }

    // 删除数据
    public void removeOfElement(int data) {
        // 获取待删除元素的下标
        int index = -1;
        for(int i = 0; i < usedSize; i++) {
            // 找到了要删除元素
            if(elem[i] == data) {
                // 获取下标
                index = i;
                break;
            }
        }
        // 没有找到
        if (index == -1) return;
        // 找到了进行删除操作
        removeOfIndex(index);
    }

    // 根据下标删除数据
    public void removeOfIndex(int index) {
        // 判断下标是否越界
        if (index < 0 || index >= usedSize) {
            throw new IndexOfArrayException("下标异常: " + index);
        }

        // 把要删除下标的后面元素前移
        for (int i = index; i < usedSize - 1; i++) {
            this.elem[i] = this.elem[i+1];
        }
        // 数组中有效长度 -1
        usedSize--;
    }

    // 删除顺序表中所有与给定数据相同的值
    public void remove(int data) {
        // 临时数组
        int[] tmp = new int[elem.length];
        int index = 0;

        // 将不同的元素放到tmp数组中
        for(int i = 0; i < usedSize; i++) {
            if (elem[i] != data) {
                tmp[index++] = elem[i];
            }
        }
        // 将删除重复元素后的数组重新赋给 elem 数组
        this.elem = tmp;
        // 重置有效元素个数
        usedSize = index;
    }

    // 获取给定值在顺序表中第一次出现的位置
    public int indexOf(int data) {
        for(int i = 0; i < usedSize; i++) {
            // 找到了
            if (elem[i] == data) {
                return i;
            }
        }
        // 没找到
        return -1;
    }

    // 获取给定值在顺序表中最后一次出现的位置
    public int lastIndexOf(int data) {
        for (int i = usedSize - 1; i >= 0; i--) {
            // 找到了
            if (elem[i] == data) {
                return i;
            }
        }
        return -1;
    }

    // 判断给定值是否在顺序表中存在
    public boolean contains(int data) {
        for (int i = 0; i < usedSize; i++) {
            // 存在
            if (elem[i] == data) {
                return true;
            }
        }
        return false;
    }

    // 获取下标处的元素
    public int get(int index) {
        if (index < 0 || index >= usedSize) {
            throw new IndexOfArrayException("下标异常: " + index);
        }
        return elem[index];
    }

    // 更新下标的元素
    public void set(int index, int data) {
        if (index < 0 || index >= usedSize) {
            throw new IndexOfArrayException("下标异常: " + index);
        }
        elem[index] = data;
    }

    // 判断顺序表是否为空
    public boolean isEmpty() {
        return usedSize == 0;
    }

    // 获取顺序表的长度
    public int size() {
        return usedSize;
    }

    // 打印顺序表的值
    public void display() {
        for (int i = 0; i < usedSize; i++) {
            System.out.print(elem[i] + " ");
        }
        System.out.println();
    }

    // 清空顺序表
    public void clear() {
        this.elem = null;
        usedSize = 0;
    }
}

Exception

java 复制代码
public class IndexOfArrayException extends RuntimeException{
    public IndexOfArrayException() {
    }

    public IndexOfArrayException(String message) {
        super(message);
    }
}

Test

java 复制代码
public class MyArrayListTest {
    public static void main(String[] args) {
        MyArrayList list = new MyArrayList();
        list.add(1);
        list.add(2);
        list.add(3);
        list.display();

        list.add(0, 111);
        list.add(3, 112);
        list.add(2, 114);
        list.display();

        list.removeOfElement(1);
        list.removeOfElement(111);
        list.removeOfElement(3);
        list.display();

        list.removeOfIndex(0);
        list.removeOfIndex(1);
        list.display();
        list.removeOfIndex(0);
        list.display();

        list.add(1);
        list.add(2);
        list.add(1);
        list.add(1);
        list.add(0);
        list.add(1);
        list.add(1);
        list.add(3);
        list.display();
        list.remove(1);
        list.display();

        list.add(2);
        list.add(0);
        list.add(2);
        list.display();
        System.out.println(list.indexOf(2));
        System.out.println(list.lastIndexOf(0));

        System.out.println(list.contains(9));
        System.out.println(list.contains(2));

        System.out.println(list.isEmpty());

        System.out.println(list.size());

        System.out.println(list.get(0));
        list.set(0, 111);
        list.display();
    }
}

通过模拟实现ArrayList后,会对底层的原理以及思想会更加理解。

相关推荐
此木|西贝3 分钟前
【设计模式】模板方法模式
java·设计模式·模板方法模式
莫有杯子的龙潭峡谷4 分钟前
4.4 代码随想录第三十五天打卡
c++·算法
luckyme_10 分钟前
leetcode 代码随想录 数组-区间和
c++·算法·leetcode
巷北夜未央12 分钟前
数据结构之二叉树Python版
开发语言·数据结构·python
wapicn9913 分钟前
手机归属地查询Api接口,数据准确可靠
java·python·智能手机·php
好好学习^按时吃饭19 分钟前
蓝桥杯2024年第十五届省赛真题-R 格式
算法·蓝桥杯
hycccccch40 分钟前
Springcache+xxljob实现定时刷新缓存
java·后端·spring·缓存
手握风云-41 分钟前
优选算法的妙思之流:分治——快排专题
数据结构·算法
熬夜苦读学习1 小时前
Linux进程信号
linux·c++·算法
白白糖1 小时前
二叉树 递归
python·算法·力扣