西河刘卡车医 发表于 2024-11-11 05:52:00

JAVA-顺序表ArrayList(实现ArrayList)

1.线性表

线性表 ( linear list ) 是n个具有相同特性的数据元素的有限序列。 线性表是一种在现实中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列...
线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构(从内存看来)并不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储。https://i-blog.csdnimg.cn/direct/c8d949663acb429a85bd8697de3a7514.png


1.1 什么是List

在聚集框架中,List是一个接口,继续自Collection。
https://i-blog.csdnimg.cn/direct/ab2a03da40f94283bf1df84079a1be9b.png
Collection也是一个接口,该接口中规范了后序容器中常用的一些方法,详细如下所示:https://i-blog.csdnimg.cn/direct/ce7801a3793f4b0c9f5ed16e123493b8.png
Iterable也是一个接口,表示实现该接口的类是可以逐个元素举行遍历的,详细如下:
https://i-blog.csdnimg.cn/direct/8b5b3a4d1c6d48ce8e63cda75e64353e.png
站在数据结构的角度来看,List就是一个线性表,即n个具有相同范例元素的有限序列,在该序列上可以执行增删 改查以及变量等操纵。

1.2 List常用方法

List常用方法:
https://i-blog.csdnimg.cn/direct/3df88b04749641758fbd94f33032c26f.png

List全部方法:
https://i-blog.csdnimg.cn/direct/6467d3a74f6d458aa1b075d162774768.png 
1.3 List的使用

注意:List是个接口,并不能直接用来实例化。
假如要使用,必须去实例化List的实现类。在聚集框架中,ArrayList和LinkedList都实现了List接口


2. 顺序表

顺序表是用一段物理地点连续(内存)的存储单元依次存储数据元素的线性结构,一样平常情况下接纳数组存储。在数组上完成 数据的增删查改。
2.1 ArrayList简介

在聚集框架中,ArrayList是一个普通的类,实现了List接口,详细框架图如下:
https://i-blog.csdnimg.cn/direct/ba746dd3969b4117a8ccd6cd16b94318.png
   【说明】
1. ArrayList是以泛型方式实现的,使用时必须要先实例化
2. ArrayList实现了RandomAccess接口,表明ArrayList支持随机访问
3. ArrayList实现了Cloneable接口,表明ArrayList是可以clone的
4. ArrayList实现了Serializable接口,表明ArrayList是支持序列化的
5. 和Vector不同,ArrayList不是线程安全的,在单线程下可以使用,在多线程中可以选择Vector或者 CopyOnWriteArrayList
6. ArrayList底层是一段连续的空间,并且可以动态扩容,是一个动态范例的顺序表
2.2 ArrayList的使用 

2.2.1 ArrayList的构造方法

ArrayList有三个构造方法
https://i-blog.csdnimg.cn/direct/909cc6bc3ebf4eb0a50dde5c909ed95a.png
无参的构造方法 :
https://i-blog.csdnimg.cn/direct/6438075d02be4adb898e273f35d16f52.png 
参数为int的构造方法:
https://i-blog.csdnimg.cn/direct/21fe328a610a4cd3a804b5ed82297590.png

泛型参数构造方法:
由于这个方法:套了很多方法欠好讲解,我就对参数和终极使用做讲解
https://i-blog.csdnimg.cn/direct/95bd6ac8e65b4f8692805d91ba07f4e9.png
https://i-blog.csdnimg.cn/direct/ab9f4a673d3647bb8caa55794253e6f0.png 使用:
https://i-blog.csdnimg.cn/direct/414161baef6842f1af1076c079d9d601.png
ArrayList保举写法:
public static void main(String[] args) {
    // ArrayList创建,推荐写法
    // 构造一个空的列表
    List<Integer> list1 = new ArrayList<>();
   
    // 构造一个具有10个容量的列表
    List<Integer> list2 = new ArrayList<>(10);
    list2.add(1);
    list2.add(2);
    list2.add(3);
    // list2.add("hello");// 编译失败,List<Integer>已经限定了,list2中只能存储整形元素
   
    // list3构造好之后,与list中的元素一致
    ArrayList<Integer> list3 = new ArrayList<>(list2);
   
    // 避免省略类型,否则:任意类型的元素都可以存放,使用时将是一场灾难
    List list4 = new ArrayList();
    list4.add("111");
    list4.add(100);
}  2.2.2 ArrayList常见方法

2.2.2.1 add或者addAll

一共有3种方法:
https://i-blog.csdnimg.cn/direct/651644a01b5b4234b07b07c26b65995d.png

https://i-blog.csdnimg.cn/direct/4aa4ded5b70b4053b5347be2bcb7c37a.png

 
2.2.2.2  removehttps://i-blog.csdnimg.cn/direct/18076e7f1a58418abbe0af711fe8efa7.png

 https://i-blog.csdnimg.cn/direct/4631374292bf406c8008b64751aebea5.png
2.2.2.3 get和set

这个很简单就不多说了
https://i-blog.csdnimg.cn/direct/94a6e2517b4040a3875eafbd2ac11944.png https://i-blog.csdnimg.cn/direct/6a21a19244454bdcbfc5e637280e914a.png
 2.2.2.4 contains

 判断是否有这个参数
https://i-blog.csdnimg.cn/direct/b75a8ce0d19f4b70ae1d65b3b74b1d60.png
https://i-blog.csdnimg.cn/direct/c3e8645ce324431e94bc2fed05d4e77c.png

2.2.2.5 indexOf和lastIndexOfhttps://i-blog.csdnimg.cn/direct/5f4450c39a904c9d9fecb51c36fee1ea.png

https://i-blog.csdnimg.cn/direct/c9c317a43bca470187820d99c3070282.png
 
2.2.2.6 subList

https://i-blog.csdnimg.cn/direct/f6f20b13b8824ad99fc1efd3999643d1.png
https://i-blog.csdnimg.cn/direct/94296834e33f4259bf717e2b8ad70ef1.png
注意:java中凡是区间都是左闭右开[ , )  也就是说上面代码只能截取到1和2下标 

2.2.2.7 clear

清空顺序表
https://i-blog.csdnimg.cn/direct/90132d1bbf9c411b82cdd1d11001b43f.png

2.3 ArrayList的遍历

public static void main(String[] args) {
      List<Integer> list = new ArrayList<>();
      list.add(1);
      list.add(2);
      list.add(3);
      list.add(4);
      list.add(5);
      //直接sout打印
      System.out.println("===sout===");
      System.out.println(list);
      // 使用下标+for遍历
      System.out.println("===for===");
      for (int i = 0; i < list.size(); i++) {
            System.out.print(list.get(i) + " ");
      }
      System.out.println();
      // 借助foreach遍历
      System.out.println("===foreach===");
      for (Integer integer : list) {
            System.out.print(integer + " ");
      }
      System.out.println();
      //迭代器
      System.out.println("===Iterator===");
      Iterator<Integer> it1 = list.iterator();
      while (it1.hasNext()) {
            System.out.print(it1.next() + " ");
      }
      System.out.println();

      System.out.println("===ListIterator===");
      //专门给List用的迭代器
      ListIterator<Integer> it2 = list.listIterator();
      while(it2.hasNext()){
            System.out.print(it2.next() + " ");
      }
      System.out.println();
      System.out.println("===ListIterator 倒着打印===");
      ListIterator<Integer> it3 = list.listIterator(list.size());
      while (it3.hasPrevious()) {
            System.out.print(it3.previous() + " ");
      }
      System.out.println();
      
    } 这里面迭代器可能大家看不懂我给大家说明一下:https://i-blog.csdnimg.cn/direct/e8b152189ea54642b108913d21c984f1.png
https://i-blog.csdnimg.cn/direct/4a448539165741f192a526521e520375.png
那hasprevious其实是判断前面是否另有?是true否false
previous向前移一位,sout并打印
 
2.4 ArratList扩容机制

ArrayList是一个动态范例的顺序表,即:在插入元素的过程中会主动扩容。以下是ArrayList源码中扩容方式:
Object[] elementData;   // 存放元素的空间
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};// 默认空间
private static final int DEFAULT_CAPACITY = 10;// 默认容量大小
public boolean add(E e) {
   ensureCapacityInternal(size + 1);// Increments modCount!!
   elementData = e;
   return true;
}
private void ensureCapacityInternal(int minCapacity) {
   ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private static int calculateCapacity(Object[] elementData, int minCapacity) {
   if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
         return Math.max(DEFAULT_CAPACITY, minCapacity);
   }
   return minCapacity;
}
private void ensureExplicitCapacity(int minCapacity) {
   modCount++;
// overflow-conscious code
   if (minCapacity - elementData.length > 0)
         grow(minCapacity);
}

private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

private void grow(int minCapacity) {
// 获取旧空间大小
    int oldCapacity = elementData.length;
// 预计按照1.5倍方式扩容
    int newCapacity = oldCapacity + (oldCapacity >> 1);
// 如果用户需要扩容大小 超过 原空间1.5倍,按照用户所需大小扩容
比特就业课
    if (newCapacity - minCapacity < 0)
       newCapacity = minCapacity;
// 如果需要扩容大小超过MAX_ARRAY_SIZE,重新计算容量大小
    if (newCapacity - MAX_ARRAY_SIZE > 0)
         newCapacity = hugeCapacity(minCapacity);
// 调用copyOf扩容
    elementData = Arrays.copyOf(elementData, newCapacity);
}
private static int hugeCapacity(int minCapacity) {
// 如果minCapacity小于0,抛出OutOfMemoryError异常
    if (minCapacity < 0)
      throw new OutOfMemoryError();

   return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE;
}
   【总结】
1. 检测是否真正必要扩容,假如是调用grow预备扩容
2. 预估必要库容的大小


[*]开端预估按照1.5倍大小扩容
[*]假如用户所需大小超过预估1.5倍大小,则按照用户所需大小扩容
[*]真正扩容之前检测是否能扩容乐成,防止太大导致扩容失败
3. 使用copyOf举行扩容
 
3. 实现顺序表

简单实现一下顺序表的大概
把要实现的方法都放在接口中:
public interface IList {
    // 新增元素,默认在数组最后新增
    void add(int data);
    // 在 pos 位置新增元素
    void add(int pos, int data);
    // 判定是否包含某个元素
    boolean contains(int toFind);
    // 查找某个元素对应的位置
    int indexOf(int toFind);
    // 获取 pos 位置的元素
    int get(int pos);
    // 给 pos 位置的元素设为 value -> 更新
    void set(int pos, int value);
    //删除第一次出现的关键字key
    void remove(int toRemove);
    // 获取顺序表长度
    int size();
    // 清空顺序表
    void clear();
    // 打印顺序表,
    // 注意:该方法并不是顺序表中的方法,为了方便看测试结果给出的
    void display();
} 在用类去实现此接口,后面全部的方法都在此类中实现:
public class MyArrayList implements IList{
    private int[] array;//存放数据
    private int size;//用来记录元素个数

    //实现一个构造方法,默认初始化为大小为10
    public MyArrayList(){
      this.array = new int;
    }
} size会默以为0,全部不消对他初始化

3.1 实现add

https://i-blog.csdnimg.cn/direct/5467625e0687464e84155c36465312ac.png
我们有size实现作为纪录元素个数的下标,当刚好可以做为尾插的位置
 https://i-blog.csdnimg.cn/direct/6e7c9e80d8ea45a58065e522294b8bc0.png
   但假如刚好满了,我们就必要扩容以是,可以用单独写一个方法判断一下https://i-blog.csdnimg.cn/direct/a0b3a564abcd40fba87b2eb11da29974.png
   @Override
    public void add(int data) {
      if(isFull()){
            //拷贝原有元素并,扩容两倍
            this.array = Arrays.copyOf(array,size * 2);
      }
      this.array = data;
      this.size++;
    }

    @Override
    public boolean isFull() {
      if(array.length == size){
            return true;
      }
      return false;
    }利用Arrays.copyOf举行扩容,他不但可以拷贝原有的元素,还可以举行扩容操纵。
 https://i-blog.csdnimg.cn/direct/30fe451a1922489a9a9e921b470a3464.png
那么在来做指定位置插入:
想要在指定位置插入,但是又不删除原位置的数据,就必须要移动
想要达到此效果,就必要从有用数据末了位置一个一个往后移动终极直接用给定的下标举行插入就可以了。
但是一旦有下标就必要举行判断下标是否合法:
    @Override
    public void add(int pos, int data) {
       try {
         checkPosOfAdd(pos);//检查pos是否合法
       }catch (PosNotLegalException e) {
         e.printStackTrace();
       }
      //移动原数组
      for (int i = size-1; i >= pos; i--) {
            this.array = this.array;
      }
      this.array = data;
      this.size++;

    }

    private void checkPosOfAdd(int pos) {
      if(pos > size || pos < 0) {
            throw new PosNotLegalException("位置不合法");
      }
    }   注意:
尾插是可以做到的,以是判断pos合法性的时间 pos == size的位置可以不做判断
异常:
public class PosNotLegalException extends RuntimeException{
    public PosNotLegalException() {

    }

    public PosNotLegalException(String mag) {
      super(mag);
    }
}
3.2 实现contains

https://i-blog.csdnimg.cn/direct/a673ec179da84c89b68166859d621f2c.png
    @Override
    public boolean contains(int toFind) {
      for (int i = 0; i < size; i++) {
            if(array == toFind) {
                return true;
            }
      }
      return false;
    }
3.3 实现indexOf 

https://i-blog.csdnimg.cn/direct/68d005cb000a4201893bd9f29f8fd7c2.png
    @Override
    public int indexOf(int toFind) {
      for (int i = 0; i < size; i++) {
            if(array == toFind) {
                return i;
            }
      }
      return -1;
    }
3.4 实现get

https://i-blog.csdnimg.cn/direct/943da8a8c5e5402bb37307cf7b5b2f64.png
    @Override
    public int get(int pos) {
      try {
            checkPosOfGetAndSet(pos);//检查pos位置是否合法
      }catch (PosNotLegalException e) {
            e.printStackTrace();
      }

      return this.array;

    }
    private void checkPosOfGetAndSet(int pos) {
      if(pos >= size || pos < 0) {
            throw new PosNotLegalException("Get/Set位置不合法");
      }
    }
3.5 实现set

https://i-blog.csdnimg.cn/direct/d8ee28ca6eaa458f937c02b348c63197.png
    @Override
    public void set(int pos, int value) {
      try {
            checkPosOfGetAndSet(pos);
      }catch (PosNotLegalException e) {
            e.printStackTrace();
      }
      array = value;
    }
3.6 实现remove

https://i-blog.csdnimg.cn/direct/d2048235f24c45599dc9d78d0571ee66.png
直接用indexOf判断是否有这个数据,有就从pos后面的数据一个个向前移动,但是末了的那个元素照旧会在原位置上,不消担心size-1就可以做到删除了,再次add数据就会覆盖到他
    @Override
    public void remove(int toRemove) {
      int pos = indexOf(toRemove);
      if(pos == -1) {
            System.out.println("没有要删除的数字");
            return;
      }


      for (int i = pos; i < size; i++) {
            array = array;
      }
      this.array = 0;
      this.size--;
    }  3.7 实现size

https://i-blog.csdnimg.cn/direct/b03af6381c6d441b8593a977a737e3ea.png
    @Override
    public int size() {
      return this.size;
    } 3.8 实现clear

https://i-blog.csdnimg.cn/direct/56c7e1d4b4ff4ba8804713d14d97ba99.png
    @Override
    public void clear() {
      for (int i = 0; i < size; i++) {
            this.array = 0;
//            this.array = null;   如果顺序表是指针类型需要全部置为空
//      因为,只要没有引用这篇空间的引用,才会释放
      }
      this.size = 0;
    } 3.9 实现display

https://i-blog.csdnimg.cn/direct/f945f9d57494407c82e4f2fe959b2513.png
    @Override
    public void display() {
      for (int i = 0; i < size; i++) {
            System.out.print(array + " ");
      }
      System.out.println();
    }

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: JAVA-顺序表ArrayList(实现ArrayList)