【数据结构与算法】模拟实现ArrayList
创始人
2025-05-30 01:21:02
0

文章目录

  • ArrayList简介
  • ArrayList模拟实现
    • 打印顺序表
    • 获取顺序表的长度
    • 判断当前的顺序表是不是满的
    • 判断当前顺序表是不是空的
    • 数组扩容
    • 新增元素,默认在数组最后新增
    • 在指定pos位置新增元素
    • 判定是否包含某个元素
    • 查找某个元素对应的位置
    • 获取指定pos位置的元素
    • 将指定pos位置的元素替换成value
    • 删除第一次出现的关键字key
    • 清空顺序表
  • 完整代码

ArrayList简介

ArrayList 类是一个可以动态修改的数组,与普通数组的区别就是它是没有固定大小的限制,我们可以添加或删除元素。ArrayList 继承了 AbstractList ,并实现了 List 接口。

简单来说ArrayList的底层就是一个动态的顺序表
而顺序表的底层就是数组.
因此要模式实现ArrayList就相当于基于数组实现一些操作,例如增删查改等。

ArrayList模拟实现

	public int[] elem;public int usedSize;//记录有效数据的个数//默认容量private static final int DEFAULT_SIZE = 10;public MyArraylist() {this.elem = new int[DEFAULT_SIZE];}

将数据放在elem这个数组中,usedSize用于计算有效数据的个数,DEFAULT_SIZE是elem的默认容量。

打印顺序表

打印顺序表,我们可以使用循环对顺序表中的数据进行打印。

    /*** 打印顺序表:* 根据usedSize判断即可*/public void display() {for (int i = 0; i < this.usedSize; i++) {System.out.print(this.elem[i] + " ");}System.out.println();}

注意:顺序表中elem的有效数据是从[0,usedSize),不能直接将整个数组打印完,我们存放的数据不一定能将数组放满。

获取顺序表的长度

此处的顺序表长度并不是底层数组的长度,而是数组种有效数据的个数.因此我们直接返回usedSize即可

	public int size() {return this.usedSize;}

判断当前的顺序表是不是满的

要想判断当前的顺序表是不是满的,只需要看看usedSize(有效数据的个数)是不是大于等与底层数组的长度 如果是满的,返回true.如果不是满的,就返回false

    /*** 判断当前的顺序表是不是满的!*/public boolean isFull() {if (size() >= this.elem.length) {return true;}return false;// 更简单的写法// return size() >= this.elem.length;}

判断当前顺序表是不是空的

判断当前顺序表是不是空的,只要看顺序表的有效数据是不是等于0即可

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

数组扩容

在实现增加元素之前,我们要考虑一下,当数组满了之后,应该怎么做.顺序表不是链表,不用考虑满不满的这种情况,但顺序表不一样,数组长度是我们事先定好的.
如果数组长度设置的过大,而数据量很少,势必会造成空间的浪费.设置的过小,可能无法把所有元素存放进去.
为了避免这种情况,我们就可以使用动态顺序表.虽然听上去很高大上,但其实也就是加了一个动态扩容的过程

扩容也并不复杂,使用Arrays.copyOf方法即可

数组名 = Arrays.copyOf(要拷贝的数组,拷贝数组的长度);

在拷贝时,如果要拷贝数组的长度超过了原数组,那么超过原数组的部分,就是数组类型的默认值

新增元素,默认在数组最后新增

在解决数组满了这种情况之后,新增元素就很简单了.
一共分为两步:首先先判断数组是否满了,数组满了,就进行扩容.然后进行新增元素.如果没满,直接增加元素即可

    /** 新增元素,默认在数组最后新增* 判断顺序表是否满了* 满了->扩容* @param data*/public void add(int data) {if (isFull()) {//扩容this.elem = Arrays.copyOf(this.elem,2*this.elem.length);}this.elem[usedSize] = data;usedSize++;}

在指定pos位置新增元素

在指定位置新增元素,首先要判断数组是否满了,如果满了要进行扩容.其次,pos位置是否合法,合法位置才能新增元素.
因为是从指定位置新增元素,如果这个位置不是在最后面,那么在新增元素时,首先要把pos及其以后位置的元素都往后移动一位,将pos位置空出来,这样才能将元素放进去

    public void add(int pos, int data) {if(isFull()){this.elem = Arrays.copyOf(this.elem,2*this.elem.length);}if(checkPos(pos)){for (int i = usedSize; i >= pos; i--) {this.elem[i] = this.elem[i-1];}this.elem[pos] = data;}this.usedSize++;}// 检查pos位置是否合法private boolean checkPos(int pos) {if (pos < 0 || pos > size()) {System.out.println("pos位置不合法");return false;}return true;//合法}

判定是否包含某个元素

直接对数组进行遍历然后比较即可,如果相等,直接返回true 如果遍历完没有与它相等的,就返回false

    // 判定是否包含某个元素public boolean contains(int toFind) {for (int i = 0; i < this.usedSize; i++) {if (toFind == this.elem[i]) {return true;}}return false;}

查找某个元素对应的位置

与上面判断是否包含某个元素相同,遍历就行.
如果相等,返回当前的i值,如果不相等,就继续遍历.如果遍历完还没找到,就返回-1

    // 查找某个元素对应的位置public int indexOf(int toFind) {for (int i = 0; i < this.usedSize; i++) {if (toFind == this.elem[i]) {return i;}}return -1;}

获取指定pos位置的元素

先检查顺序表是否为空,如果是空的.直接返回-1.如果不为空,再检查pos位置是否合法.
用自定义异常解决了顺序表为空这种情况,处理方法不一,合理即可

    // 获取 pos 位置的元素public int get(int pos) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(checkPos(pos)){return this.elem[pos];}return -1;}

将指定pos位置的元素替换成value

先判断顺序表是否为空,在判断pos位置是否合法.满足即可将pos位置的值更新为value

    // 给 pos 位置的元素设为【更新为】 valuepublic void set(int pos, int value) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(pos < 0 || pos > size()){System.out.println("pos位置不合法");return;}this.elem[pos] = value;}

删除第一次出现的关键字key

先判断顺序表是否为空,是空直接返回
然后判断顺序表中是否存在key,如果存在key,直接让key所在的位置的后面元素,一个一个往前覆盖,如果不存在,直接返回即可

    public void remove(int key) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(contains(key)) {int ret = indexOf(key);for (int i = ret; i < usedSize; i++) {this.elem[i] = this.elem[i+1];}}this.usedSize--;}

清空顺序表

public void clear() {this.usedSize = 0;}

这里的清空并不是真的清空,而是下次增加元素时,会把之前有的元素覆盖掉.如果不覆盖,那么原来的数据还是存在数组中的
如果想要彻底清空顺序表,就需要使用循环,将数组一个一个置为0

完整代码

主要部分:

import java.util.Arrays;public class MyArrayList {public int[] elem;public int usedSize;//0//默认容量private static final int DEFAULT_SIZE = 10;public MyArrayList() {this.elem = new int[DEFAULT_SIZE];}/*** 打印顺序表:* 根据usedSize判断即可*/public void display() {for (int i = 0; i < this.usedSize; i++) {System.out.print(this.elem[i] + " ");}System.out.println();}/** 新增元素,默认在数组最后新增* 判断顺序表是否满了* 满了->扩容* @param data*/public void add(int data) {if (isFull()) {//扩容this.elem = Arrays.copyOf(this.elem,2*this.elem.length);}this.elem[usedSize] = data;usedSize++;}/*** 判断当前的顺序表是不是满的!* @return true:满   false代表空*/public boolean isFull() {
//        if (size() >= this.elem.length) {
//            return true;
//        }
//        return false;return size() >= this.elem.length;}private boolean checkPos(int pos) {if (pos < 0 || pos > size()) {System.out.println("pos位置不合法");return false;}return true;//合法}/** 在 pos 位置新增元素* 方法:将pos位置的元素向后移动* 情况:*  pos位置不合法*  不能间隔* @param pos* @param data**/public void add(int pos, int data) {if(isFull()){this.elem = Arrays.copyOf(this.elem,2*this.elem.length);}if(checkPos(pos)){for (int i = usedSize; i >= pos; i--) {this.elem[i] = this.elem[i-1];}this.elem[pos] = data;}this.usedSize++;}// 判定是否包含某个元素public boolean contains(int toFind) {for (int i = 0; i < this.usedSize; i++) {if (toFind == this.elem[i]) {return true;} else {continue;}}return false;}// 查找某个元素对应的位置public int indexOf(int toFind) {for (int i = 0; i < this.usedSize; i++) {if (toFind == this.elem[i]) {return i;}}return -1;}// 获取 pos 位置的元素public int get(int pos) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(checkPos(pos)){return this.elem[pos];}return -1;}//判断顺序表是否为空private boolean isEmpty() {return size()==0;}// 给 pos 位置的元素设为【更新为】 valuepublic void set(int pos, int value) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(pos < 0 || pos > size()){System.out.println("pos位置不合法");return;}this.elem[pos] = value;}/*** 删除第一次出现的关键字key* @param key*/public void remove(int key) {if(isEmpty()){throw new EmptyException("顺序表为空!");}if(contains(key)) {int ret = indexOf(key);for (int i = ret; i < usedSize; i++) {this.elem[i] = this.elem[i+1];}}this.usedSize--;}// 获取顺序表长度public int size() {return this.usedSize;}// 清空顺序表public void clear() {this.usedSize = 0;}
}

自定义异常部分

public class EmptyException extends RuntimeException{public EmptyException() {}public EmptyException(String message) {super(message);}
}

相关内容

热门资讯

银行、消金公司助贷余额增速不得... 近日,中国证券报记者从多位业内人士处独家获悉,5月以来,多地金融监管部门对部分中小银行、消金公司下达...
朱鸿接任陈航,担任钉钉科技有限... 消费日报-今朝新闻讯 天眼查显示,6月23日,钉钉科技有限公司发生工商变更,陈航卸任法定代表人、董事...
3日累跌超20%,德创环保:公... 6月25日, 德创环保(603177.SH)公告,公司股票于2026年6月23日、6月24日和6月2...
北京发布2026年第七轮拟供商... 央广网北京6月25日消息(记者门庭婷)6月25日,北京市规划和自然资源委员会网站发布了2026年第七...
开放麦 | 启明创投胡奇:从A... “2026年,创投圈的浪潮再次翻涌:AI从技术概念走进产业深水区,硬科技创业从“小众赛道” 变成“主...
腾讯孙忠怀:在行业转身处 6月24日,2026腾讯视频年度发布在上海举行。腾讯公司副总裁、腾讯在线视频董事长孙忠怀以《在行业转...
加息,突变!美联储,重磅传来!... 美联储政策路径突生变数。 美国商务部经济分析局最新公布的数据显示,5月个人消费支出(PCE)物价指数...
6月合肥上门收金必看!5步避坑... 2026年6月,合肥黄金市场持续高位运行,不少市民翻出家里闲置的旧金饰、投资金条想变现,上门回收因为...
潮汕女富豪挂帅后加码液冷!祥鑫... 潮汕女强人,带着百亿公司加码液冷散热。 6月24日晚间,祥鑫科技(002965.SZ)公告称,公司董...
马斯克向太空要电,GobiX ... 一场关于「去哪里找电」的全球竞赛,正在朝两个方向展开。 作者|周永亮 编辑| 郑玄 「太空光伏是不是...
原料药行业陷入周期低谷 有药企... 每经记者|许立波 每经编辑|魏文艺 “过完年到现在,我们整个团队每个月都在出差,跑遍了亚非拉、欧美市...
家门口筛查白内障!永顺泽家镇暖... 大众卫生报·新湖南客户端6月25日讯(通讯员 彭雪姣)为切实解决辖区老年性白内障患者异地就医奔波、就...
终于等到!油价马上再大跌,这个... 点击添加图片描述(最多60个字) 编辑 各位车主朋友,好消息接二连三! 继6月18日油价大幅下调...
丈量出海新路 世界酒庄影响力指... 长期以来,全球酒庄评价体系由西方机构主导,且大多局限于单一酒种、单一评价维度,这一局面正逐渐被打破。...
峰瑞资本创始合伙人李丰:从资本... “2026年,创投圈的浪潮再次翻涌:AI从技术概念走进产业深水区,硬科技创业从“小众赛道” 变成“主...
原创 A... 迈向成熟,还有茁壮成长的机会。 作者 | 方璐 编辑丨于婞 来源 | 野马财经 2026年6月21日...
为企业解锁出海新通道!亚太中小... 6月24日下午,作为2026年APEC中小企业工商论坛的重要组成部分,亚太中小企业国际化合作发展论坛...
君赛生物港股IPO,增聘兴证国... 跟丰宜科技一样,正冲刺港股IPO的上海君赛生物股份有限公司(简称“君赛生物”)增聘一位整体协调人。 ...
圣邦股份明日上市:暗盘涨24%... 雷递网 雷建平 6月25日 圣邦微电子(北京)股份有限公司(简称:“圣邦股份”,股票代码:“0366...
科技“吃肉”,券商跟着“喝汤”... 当科技持续成为市场核心主线,押中硬科技项目的券商也成为被追逐的焦点。 6月24日,半导体零部件概念股...