ArrayList简介

ArrayList是一个动态数组,Array的复杂版本,它提供了动态的增加和减少元素,实现了ICollection和IList接口,灵活的设置数组的大小等好处,ArrayList不是线程安全的。

MyArrayList

在实现MyArrayList前先实现MyList,MyAbstractList,他们的关系如下。

MyArrayList--->(继承于)MyAbstractList--->(实现)MyList接口--->(实现)java.lang.Iterable接口

通过编写新的ArrayList来实现java中ArrayList的部分功能

MyList接口定义如下

public interface MyList<E> extends Iterable<E> {

    public void add(E e);
public void add(int index, E e);
public void clear();
public boolean contains(E e);
public E get(int index);
public int indexOf(E e);
public boolean isEmpty();
public int lastIndexOf(E e);
public boolean remove(E e);
public E remove(int index);
public E set(int index, E e);
public int size();
}

MyAbstractList类定义如下

public abstract class MyAbstractList<E> implements MyList<E> {

    protected int size = 0;

    protected MyAbstractList() {

    }

    protected MyAbstractList(E[] objects) {
for(int i = 0; i < objects.length; i++) {
add(objects[i]);
}
} @Override
public void add(E e) {
add(size, e);
} @Override
public boolean isEmpty() {
return size == 0;
} @Override
public int size() {
return size;
} @Override
public boolean remove(E e) {
if(indexOf(e) >= 0) {
remove(indexOf(e));
return true;
}else {
return false;
}
}
}

MyArrayList定义如下

import java.util.Iterator;

public class MyArrayList<E> extends MyAbstractList<E> {
public static final int INITIAL_CAPACITY = 16;//初始容量
private E[]data = (E[])(new Object[INITIAL_CAPACITY]);; public MyArrayList() {
} public MyArrayList(int length) {
if(length >= 0) {
data = (E[])(new Object[length]);
}
} public MyArrayList(E[] objects) {
for(int i = 0; i < objects.length; i++) {
add(objects[i]);
}
} @Override
public void add(int index, E e) {
// TODO Auto-generated method stub
ensureCapacity(); for(int i = size - 1; i >= index; i--) {//将data[index]和后面的元素往后移动一位
data[i + 1] = data[i];
} data[index] = e;
size++;
} //判断容量满时,新建一个容量为原来两倍+1的数组
private void ensureCapacity() {
if(size >= data.length) {
E[] newData = (E[])(new Object[size * 2 + 1]);//java源码扩容1.5倍+1
System.arraycopy(data, 0, newData, 0, size);
data = newData;
}
} @Override
public void clear() {
// TODO Auto-generated method stub
data = (E[])(new Object[INITIAL_CAPACITY]);
size = 0;
} @Override
public boolean contains(E e) {
// TODO Auto-generated method stub
for(int i = 0; i < size; i++) {
if(e.equals(data[i])) {
return true;
}
}
return false;
} @Override
public E get(int index) {
// TODO Auto-generated method stub
checkIndex(index);//判断下标是否合法
return data[index];
} private void checkIndex(int index) {
if(index < 0 || index >= size) {
throw new IndexOutOfBoundsException("index " + index + " out of bounds");
}
} @Override
public int indexOf(E e) {
// TODO Auto-generated method stub
for(int i = 0; i < size; i++) {
if(e.equals(data[i])) {
return i;
}
}
return -1;
} @Override
public int lastIndexOf(E e) {
// TODO Auto-generated method stub
for(int i = size - 1; i >= 0; i--) {
if(e.equals(data[i])) {
return i;
}
}
return -1;
} @Override
public E remove(int index) {
// TODO Auto-generated method stub
checkIndex(index);//判断下标是否合法
E e = data[index];//获取删除对象的值 for(int i = index; i < size - 1; i++) {//将data[index]后面的元素往前移动一位
data[i] = data[i+1];
} data[size] = null;//删除最后的元素
size--;
return e;
} @Override
public E set(int index, E e) {
// TODO Auto-generated method stub
checkIndex(index);//判断下标是否合法
E old = data[index];
data[index] = e;
return old;
} //重写java.lang.Object.toString方法,打印整个数组
@Override
public String toString() {
StringBuilder result = new StringBuilder("["); for(int i = 0; i < size; i++) {
result.append(data[i]);
if(i < size - 1) {
result.append(",");
}
} return result + "]";
} //新建一个刚好等于size的数组,即将数组容量设为刚好足够存放的大小
public void trimToSize() {
if(size != data.length) {
E[] newData = (E[])(new Object[size]);
System.arraycopy(data, 0, newData, 0, size);
data = newData;
}
} @Override
public Iterator<E> iterator() {
// TODO Auto-generated method stub
return new ArrayListIterator();
} private class ArrayListIterator implements Iterator<E> {
private int current = 0; @Override
public boolean hasNext() {
// TODO Auto-generated method stub
return (current < size);
} @Override
public E next() {
// TODO Auto-generated method stub
return data[current++];
}
}
}

测试代码如下

public class TestMyArrayList {

    public static void main(String[] args) {
// TODO Auto-generated method stub // String[] ss = {"China","America","Germany","Canada","France","Germany"};
MyList<String> list = new MyArrayList<String>();
System.out.println("测试add(E e)方法和add(int index, E e)方法添加数据。");
list.add("America");
list.add(0,"China");
list.add("Canada");
list.add("France");
list.add(2,"Germany");
list.add("Germany");
System.out.println("通过toString()方法打印数组" + list.toString());
System.out.println("通过size()方法获取长度:" + list.size());
System.out.println("测试contains()方法:" + list.contains("France"));
System.out.println("通过get()方法获取指定下标的值:" + list.get(1));
System.out.println("通过indexOf()方法获取指定值的下标:" + list.indexOf("Germany"));
System.out.println("通过lastIndexOf()方法获取最后出现的指定值的下标:" + list.lastIndexOf("Germany"));
System.out.println("通过下标删除指定值:" + list.remove(5));
System.out.println("删除后的数组:" + list.toString());
System.out.println("通过set()方法修改值:" + list.set(2, "Russia"));
System.out.print("通过foreach遍历数组:");
for(String s:list) {
System.out.print(s.toUpperCase() + " ");
}
} }

测试结果

思考:为什么MyArrayList类中的ensureCapacity()扩容倍数是2 * x + 1而不是2 * x?

因为如果我们创建了一个长度为0的数组,如果扩容倍数是2 * x,那么这个数组就无法扩容了。

实现MyArrayList类深入理解ArrayList的更多相关文章

  1. 理解ArrayList与LinkedList的区别

    一.先来看看ArrayList与LinkedList 在JDK中所在的位置 从图中可以看出,ArrayList与LinkedList都是List接口的实现类,因此都实现了List的所有未实现的方法,只 ...

  2. 深入理解ArrayList与LinkedList的区别

    一.先来看看ArrayList与LinkedList 在JDK中所在的位置 从图中可以看出,ArrayList与LinkedList都是List接口的实现类,因此都实现了List的所有未实现的方法,只 ...

  3. [BS-18] 对OC中不可变类的理解

    对OC中不可变类的理解 OC中存在很多不可变的类(如NSString,NSAttributedString,NSArray,NSDictionary,NSSet等),用它们创建的对象存在于堆内存中,但 ...

  4. 编写测试类,了解ArrayList的方法

    这篇文章主要介绍了C#中动态数组用法,实例分析了C#中ArrayList实现动态数组的技巧,非常具有实用价值,需要的朋友可以参考下 本文实例讲述了C#中动态数组用法.分享给大家供大家参考.具体分析如下 ...

  5. 16、Collection接口及其子接口Set和List(常用类LinkedList,ArrayList,Vector和Stack)

    16.Collection接口 Collection是最基本的集合接口,一个Collection代表一组Object,即Collection的元素(Elements).一些Collection允许相同 ...

  6. 对Java中properties类的理解

    转载于:https://www.cnblogs.com/bakari/p/3562244.html 一.Java Properties类 Java中有个比较重要的类Properties(Java.ut ...

  7. AtomicInteger类的理解与使用

    AtomicInteger类的理解与使用 首先看两段代码,一段是Integer的,一段是AtomicInteger的,为以下: public class Sample1 { private stati ...

  8. FileChannel类的理解和使用

    FileChannel类的理解和使用(java.nio.channels.FileChannel) 知识点: 1.FileChannel类及方法理解:2.普通输入输出流复制文件:3.FileChann ...

  9. 从源代码来理解ArrayList和LinkedList差别

    从源代码理解ArrayList和LinkedList差别 ArrayList ArrayList默认容量为10,实质是一个数组用于存放元素,size表示ArrayList所包括的元素个数. Array ...

随机推荐

  1. 类装载器DexClassLoader (android内核剖析)

    在java环境中,有个概念叫做"类装载器",其作用是动态装载Class文件.标准的java SDK中有一个ClassLoader类,借助它可以装载 想要的Class文件,每个Cla ...

  2. javascript语法之number对象和Math对象

    这两个对象很简单,一个例子就能掌握用法. 一:number对象. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional// ...

  3. 敏捷测试(1)--TDD概念

    题记 本系列笔记将从测试人员的角度,总结在百度两年来的测试经验,记录一个完整的基于敏捷流程的验收测试全过程,分享在测试过程中的一些知识和经验,以及自己的一些理念.总结自己,也希望对大家有益. 概念 验 ...

  4. Leetcode_162_Find Peak Element

    本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/43415313 A peak element is an e ...

  5. 使用 /sys 文件系统访问 Linux 内核

    sysfs 与 /sys sysfs 文件系统总是被挂载在 /sys 挂载点上.虽然在较早期的2.6内核系统上并没有规定 sysfs 的标准挂载位置,可以把 sysfs 挂载在任何位置,但较近的2.6 ...

  6. 如何手动实现C语言中的字符串操作

    学了字符串操作,很多人也许学了大概知道怎么用,但是太久没用就忘了,恰恰这是找软件工程师或者嵌入式工程师以及C,C++相关的笔试面试必考的题目!接下来我们来看看如何手动实现这些相关的函数. 废话不多说, ...

  7. 网络I/O中的同步、异步、阻塞和非阻塞概念

    在学习网络编程过程中,经常会把这几个概念搞混淆. 同步I/O与异步I/O区别 我们先来看一下操作I/O时涉及的对象和步骤(这里我们以read为例): 这里会涉及到两个系统对象,一个是调用这个I/O的应 ...

  8. Android实现RecyclerView侧滑删除和长按拖拽-ItemTouchHelper

    RecyclerView这个被誉为ListView和GirdView的替代品,它的用法在之前的一篇博文中就已经讲过了,今天我们就来实现RecyclerView的侧滑删除和长按拖拽功能,实现这两个功能我 ...

  9. BT币(金融有风险,投资需谨慎)哥的失败投资

    谁都知道bt币是一个旁氏骗局, 而进去的人,就必须保证自己不赔钱,所以只能随着大潮往前走,谁也不能让它跌 压垮骆驼的最后一根稻草, 还是幕后有个 推手, 在炒作 BT币, 事实上,作为新的投资项目,B ...

  10. lpad函数

    函数介绍 lpad函数是Oracle数据库函数,lpad函数从左边对字符串使用指定的字符进行填充.从其字面意思也可以理解,l是left的简写,pad是填充的意思,所以lpad就是从左边填充的意思. 2 ...