动态数组:扩容与缩容

动态数组:扩容与缩容

动态数组 是一种动态存储的线性表,所有元素的内存地址都是连续的。

很多语言的开发都需要用到数组来存储数据,本文主要是学习了一下实现数组的一些基本方法,以及扩容操作和缩容操作的原理。

数组打印
Java中数组的打印直接是打印的对象的类型,例如直接打印:

     ArrayList list  = new ArrayList();
     list.add(11);
     list.add(22);
     list.add(33);
     System.out.println(list);

打印结果:

com.yy.ArrayList@4e25154f

为了达到打印能看到数组内部的元素,或者其他情况,需要重写ArrayList中的toString()方法:

   @Override
    public String toString() {
        StringBuilder string = new StringBuilder();
        string.append("size=").append(size).append(", [");
        for (int i = 0; i < size; i++) {
            if (i != 0) string.append(", ");
            string.append(elements[i]);
        }
        string.append("]");
        return string.toString();
    }

重写以后,打印结果如下:

size=3, [11, 22, 33]

快速生成代码:

  • 快速生成构造函数:command + option + s -> Generate Contructor using Fields
  • 快速生成重写toString方法:command + option + s -> Generate toString...
image.png

示例:一个person类生成的构造方法和toString:

public class Person {
    private int age;
    private String name;
    public Person(int age, String name) {
        this.age = age;
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person [age=" + age + ", name=" + name + "]";
    }

}

1.1 添加元素

1.1.1 添加元素到末尾

添加元素到末尾思路:

  • 当 size == 0 的时候,elements[0] = element;
  • 当 size > 0 的时候,elements[size] = element;
  • 最后size ++ ;
    /*
     *  添加元素到末尾
     **/
    public void add(int element) {
        elements[size ++] = element;
    }

1.1.2 删除index位置元素

删除元素-摘自小码哥课件

如图,在index=3位置,未删除之前elements[3] = 44;
删除index == 3位置元素44 思路:

  • index == 5 6 7 位置元素向前挪动一位(for 循环)
  • index == 6位置元素不需要处理,因为删除完成以后size --,只能访问到前6个元素。
    /*
     *  删除index位置元素
     **/
    public int remove(int index) {
        if (index < 0 || index >= size) {
            throw new IndexOutOfBoundsException("Index" + index + ",Size"+ size);
        }
        int old = elements[index];
        for (int i = index + 1; i < size; i++) {
            elements[i - 1] = elements[i];
        }
        size --;
        return old;
    }

1.1.3 添加元素到index位置

image.png

如图,在index=2位置,添加一个新的元素;
原有index == 2,3,4位置元素往后挪动 思路:

  • index == 2 3 4 位置元素向后挪动一位(for 循环)
  • 注意: 挪动顺序,从高往低
    • 4-->5;
    • 3-->4;
    • 2-->3;
  • 插入新元素到index == 2。
  • size ++;
       /*
        *  添加,插入元素
        **/
       public void add(int index, int element) {
           if (index < 0 || index > size) {
               throw new IndexOutOfBoundsException("Index" + index + ",Size"+ size);
           }
           for (int i = size - 1 ;i >= index; i--) {
               elements[i + 1] = elements[i];
           }
           elements[index] = element;
           size ++;
       }

2. 数组的扩容

当添加元素时,创建的elements数组容量不够了,这时候不能直接给elements扩容,因为创建的时候elements的容量就已经开辟好了;

所以需要创建一个新的数组,并将elements中的元素挪到新的数组中;

步骤如下:

  • 判断容量是否够用,也就是size + 1 <= elements.length;
  • 创建一个新的数组newElements,容量为elements的1.5倍(使用位运算效率更高);
  • 将elements中的元素挪到newElements中;
  • 将element指针指向newElements;
扩容示例-摘自小码哥课件

代码实现如下:

    /*
     * 扩容
     * */
    private void ensureCapacity(int capacity){
        if(elements.length >= capacity) return;
        int oldCapacity = elements.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        int[] newElements = new int[newCapacity];
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[i];
        }
        elements = newElements;
        System.out.println(oldCapacity + "扩容到" + newCapacity);
    }

测试是否能扩容实现:


测试扩容

2. 数组的缩容

当删除元素时,elements数组中元素会越来越少,当少到一定程度时,数组原有开辟的空间就会浪费,所以就要考虑到缩容操作;

    /*
     *  删除index位置元素
     **/
    public E remove(int index) {
        rangeCheck(index);
        E old = elements[index];
        for (int i = index + 1; i < size; i++) {
            elements[i - 1] = elements[i];
        }
        elements[--size] = null; //将最后一个元素存放的指针清空
        trim(); //缩容操作
        return old;
    }

所以和扩容一样,需要创建一个新的数组,并将elements中的元素挪到新的数组中;

步骤如下:

  • 判断容量是否够用,也就是size 大于elements容量的一半时,或者容量小于DEFAULIT_CAPACITY默认容量10时,没必要进行缩容操作;
  • 创建一个新的数组newElements,容量为elements的1/2(使用位运算效率更高);
  • 将elements中的元素挪到newElements中;
  • 将element指针指向newElements;
  private void trim(){
      int capacity = elements.size;
      int newCapacity = capacity >> 1; //右移一位相当于除以2
      if(size >= newCapacity || capacity <= DEFAULIT_CAPACITY) return;
      @SuppressWarnings("unchecked")
      E[] newElements = (E[]) new Object[newCapacity];
      for (int i = 0; i < size; i ++){
          newElements[i] = elements[i];
      }
     elements = newElements;
     System.out.println(capacity + "缩容到" + newCapacity);
  }
image.png

2 泛型

在之前实现的动态数组中,存放的元素只能是int类型,有一定的局限性;
所以我们要将数组的内容改成可以存放对象,这样里面存放任何东西都行;包括基本数据类型和对象类型;

java中存在泛型这个概念:

2.1 Java 泛型(generics)

是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。
泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。

2.2 泛型方法

你可以写一个泛型方法,该方法在调用时可以接收不同类型的参数。根据传递给泛型方法的参数类型,编译器适当地处理每一个方法调用。
下面是定义泛型方法的规则:

  • 所有泛型方法声明都有一个类型参数声明部分(由尖括号分隔),该类型参数声明部分在方法返回类型之前(在下面例子中的)。
  • 每一个类型参数声明部分包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符。
  • 类型参数能被用来声明返回值类型,并且能作为泛型方法得到的实际参数类型的占位符。
  • 泛型方法体的声明和其他方法一样。注意类型参数只能代表引用型类型,不能是原始类型(像int,double,char的等)。
public class GenericMethodTest
{
   // 泛型方法 printArray                         
   public static < E > void printArray( E[] inputArray )
   {
      // 输出数组元素            
         for ( E element : inputArray ){        
            System.out.printf( "%s ", element );
         }
         System.out.println();
    }
 
    public static void main( String args[] )
    {
        // 创建不同类型数组: Integer, Double 和 Character
        Integer[] intArray = { 1, 2, 3, 4, 5 };
        Double[] doubleArray = { 1.1, 2.2, 3.3, 4.4 };
        Character[] charArray = { 'H', 'E', 'L', 'L', 'O' };
 
        System.out.println( "整型数组元素为:" );
        printArray( intArray  ); // 传递一个整型数组
 
        System.out.println( "\n双精度型数组元素为:" );
        printArray( doubleArray ); // 传递一个双精度型数组
 
        System.out.println( "\n字符型数组元素为:" );
        printArray( charArray ); // 传递一个字符型数组
    } 
}

2.3 修改动态数组存放泛型对象

  1. 修改类型参数声明;

  2. 修改数组elements中存放的元素类型int为E;

    • 在java中所有的类最终都继承自 java.lang.Object

    private E[] elements = (E[]) new Object[capacity];

    • 点击修改警告,会添加类型转换,
  3. 修改扩容时newElements中的存放元素类型int为E,以及其他传入的参数,返回的参数为int型的为E(注意传入的参数index不需要修改)

修改后的打印:


image.png

注意:修改后,创建数组时要指定类型,比如Person,Integer,Double,Integer相当于Int的对象类型;

2.4 数组存放int和泛型的区别

  1. 当数组中存放的是基本数据类型时,存放的就是数据本身;
  2. 当数组中存放的是泛型对象时,直接将对象放到数组中不现实,因为每个对象的属性值数不一样,所以对象的大小不一样,会导致数组分配的内存特别大;
  3. 所以对象类型的元素,存放到数组中的是对象的指针,如下图:
image.png

当查询数组中的某一个元素时,返回的是这个对象的地址,再通过地址去查找这个对象;

2.4.1 内存管理细节:数组的清空操作

1、在动态数组的clear清空操作时,要特殊处理:

    /*
     * 清除元素
     * **/
    public void clear(){
        //清空数组内部保存的对象的地址 --> 及时的清除不要的对象
        for (int i = 0; i < size; i++) {
            elements[i] = null;
        }
        size = 0;
    }

清空数组时,需要遍历一遍将对应的地址置为null,地址置为null后就回自动将对象释放,不然得等数组中重新存放元素时,这个对象才能释放,万一不存放元素,这个对象就会一直在内存中;

注意:不能将elements = null,也就是objects指针置为空,这样操作也能达到目的,但是以后只要是清空了,数组也会被释放,每次都得重新创建数组,这不合理,合理的操作应该是只需要将数组中存放的地址置为空就行。

2、 在数组remove移除操作时,挪动完成以后对最后一个位置存放的元素的指针应当清除操作:

/*
*  删除index位置元素
**/
public E remove(int index) {
   rangeCheck(index);
   E old = elements[index];
   for (int i = index + 1; i < size; i++) {
       elements[i - 1] = elements[i];
   }
   elements[--size] = null; //将最后一个元素存放的指针清空
   return old;
}

2.5 数组是否可以存放null

1、 假如数组不允许存放null,在add操作时,加入判断即可:

if(element == null) return;

2、但是java官方默认是可以加入null的,但是在返回index方法需要特殊处理:
index 返回第一个null元素的index;

    /*
     * 元素索引
     * ***/
    public int indexof(E element) {
        if (element == null) {
            for (int i = 0; i < size; i++) {
                if(elements[i]==null) return i;
            }
        }else{
            for (int i = 0; i < size; i++) {
                if(element.equals(elements[i])) return i;
            }   
        }
        return ELEMENT_NOT_FOUND;
    }

你可能感兴趣的:(动态数组:扩容与缩容)