集合

Collection

  • List,Queue,Set都是继承Collection接口;
    • Set:Set不允许包括重复元素,其他的和collection相同;
    • List:代表一个元素有序、可重复的集合,集合中每个元素都有其对应的顺序索引。List集合允许使用重复元素,可以通过索引来访问指定位置的集合元素 。
    • List集合:代表一个元素有序、可重复的集合,集合中每个元素都有其对应的顺序索引。List集合允许使用重复元素,可以通过索引来访问指定位置的集合元素 。队列的头部是在队列中存放时间最长的元素,队列的尾部是保存在队列中存放时间最短的元素。新元素插入(offer)到队列的尾部,访问元素(poll)操作会返回队列头部的元素。通常,队列不允许随机访问队列中的元素。
  • Collection定义了多种方法供子类调用。
    • 添加:add,addAll
    • 删除:clear(),remove(),removeAll()
boolean add(E e)Ensures that this collection contains the specified element (optional operation).
boolean addAll(Collection<? extends E> c)Adds all of the elements in the specified collection to this collection (optional operation).
void clear()Removes all of the elements from this collection (optional operation).
boolean contains(Object o)Returns true if this collection contains the specified element.
boolean containsAll(Collection<?> c)Returns true if this collection contains all of the elements in the specified collection.
boolean equals(Object o)Compares the specified object with this collection for equality.
int hashCode()Returns the hash code value for this collection.
boolean isEmpty()Returns true if this collection contains no elements.
Iterator<E> iterator()Returns an iterator over the elements in this collection.
default Stream<E> parallelStream()Returns a possibly parallel Stream with this collection as its source.
boolean remove(Object o)Removes a single instance of the specified element from this collection, if it is present (optional operation).
boolean removeAll(Collection<?> c)Removes all of this collection's elements that are also contained in the specified collection (optional operation).
default boolean removeIf(Predicate<? super E> filter)Removes all of the elements of this collection that satisfy the given predicate.
boolean retainAll(Collection<?> c)Retains only the elements in this collection that are contained in the specified collection (optional operation).
int size()Returns the number of elements in this collection.
default Spliterator<E> spliterator()Creates a Spliterator over the elements in this collection.
default Stream<E> stream()Returns a sequential Stream with this collection as its source.
Object[] toArray()Returns an array containing all of the elements in this collection.
<T> T[] toArray(T[] a)Returns an array containing all of the elements in this collection; the runtime type of the returned array is that of the specified array.

List

wiki

ArrayList

学习要点

  • 列表的增,删,改,查的各种方法;
  • 列表遍历,扩容的机制;

ArrayList的结构图

  • ArrayList 继承了AbstractList,实现了List。它是一个数组队列,提供了相关的添加、删除、修改、遍历等功能。

  • ArrayList 实现了RandmoAccess接口,即提供了随机访问功能。RandmoAccess是java中用来被List实现,为List提供快速访问功能的。

  • 在ArrayList中,我们即可以通过元素的序号快速获取元素对象;这就是快速随机访问。稍后,我们会比较List的“快速随机访问”和“通过Iterator迭代器访问”的效率。

  • ArrayList 实现了Cloneable接口,即覆盖了函数clone(),能被克隆。

  • ArrayList 实现java.io.Serializable接口,这意味着ArrayList支持序列化,能通过序列化去传输。

  • 和Vector不同,ArrayList中的操作不是线程安全的!所以,建议在单线程中才使用ArrayList,而在多线程中可以选择Vector或者CopyOnWriteArrayList。

ArrayList的定义

Java ArrayList的自动扩容机制

public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable
        {

        }

what:ArrayList的本质-ArrayList的构造函数

  • ArrayList是用数组实现,元素放在elementData数组中;
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
transient Object[] elementData;

public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; // 返回一个空数组的引用
}

public ArrayList(int initialCapacity) { //如果指定List长度,返回对应长度的Object数组
    if (initialCapacity > 0) {
        this.elementData = new Object[initialCapacity];
    } else if (initialCapacity == 0) {
        this.elementData = EMPTY_ELEMENTDATA;
    } else {
        throw new IllegalArgumentException("Illegal Capacity: "+
                                           initialCapacity);
    }
}

public ArrayList(Collection<? extends E> c) { //使用其他的集合作为参数
    elementData = c.toArray();
    if ((size = elementData.length) != 0) {
        // c.toArray might (incorrectly) not return Object[] (see 6260652)
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    } else {
        // replace with empty array.
        this.elementData = EMPTY_ELEMENTDATA;
    }
}

ArrayList扩容机制:ensureCapacityInternal()

private void ensureCapacityInternal(int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
    }

    ensureExplicitCapacity(minCapacity);
}

如果数组为空数组,则将数组扩充为DEFAULT_CAPACITY = 10大小的数组

private void ensureExplicitCapacity(int minCapacity) {
    modCount++;

    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

在扩充数组的时候,modCound会自动加1;

private void grow(int minCapacity) {
    // overflow-conscious code
    int oldCapacity = elementData.length;
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    if (newCapacity - MAX_ARRAY_SIZE > 0)
        newCapacity = hugeCapacity(minCapacity);
    // minCapacity is usually close to size, so this is a win:
    elementData = Arrays.copyOf(elementData, newCapacity);
}

可以看出,新的容量会是max[1.5*oldCapacity, minCapacity],容量最大为Integer.MAX_VALUE

总结:扩容时modCount自动加1,新的容量是max[1.5*oldCapacity, minCapacity],容量的最大值为Integer.MAX_VALUE

遍历集合的方式

  • 通过Iterator遍历

    public static void main(String[] args)
    {
        List<String> list = new ArrayList<>();
        list.add("uranus");
        list.add("leon");
        list.add("test");
    
        Iterator iter = list.iterator();
    
        while(iter.hasNext())
        {
            System.out.println(iter.next());
        }
    }
    
  • 通过foreach方式

    public static void main(String[] args)
    {
        List<String> list = new ArrayList<>();
        list.add("uranus");
        list.add("leon");
        list.add("test");
    
        for (String str : list)
        {
            System.out.println(str);
        }
    }
    
  • 通过索引方式

    public static void main(String[] args)
    {
        List<String> list = new ArrayList<>();
        list.add("uranus");
        list.add("leon");
        list.add("test");
    
        for (int i = 0; i < list.size(); i++)
        {
            System.out.println(list.get(i));
        }
    }
    

foreach遍历本质和Iterator遍历是一样的,通过编译后的代码可以看出

// foreach 方式编译结果
public static void main(String args[])
{
    List list = new ArrayList();
    list.add("uranus");
    list.add("leon");
    list.add("test");
    String str;
    for(Iterator iterator = list.iterator(); iterator.hasNext(); System.out.println(str))
        str = (String)iterator.next();
}

拷贝集合:浅拷贝和深拷贝

  • clone()方法的误区
List<String> list1 = new ArrayList<>();
ArrayList list2 = (ArrayList) list.clone(); //编译报错

ArrayList<String> list3 = new ArrayList<>();
ArrayList list4 = (ArrayList) list.clone(); //编译不会报错

list1指向的是ArrayList类型的实例,为什么调用clone的时候不是调用ArrayList中的clone方法,而是在List中寻找clone方法,导致报错?

Error:(11, 43) java: 找不到符号
  符号:   方法 clone()
  位置: 类型为java.util.List<java.lang.String>的变量 list
  • 浅拷贝

    public class ListBasic {
        public static void main(String[] args)
        {
            List<Student> list = new ArrayList<>();
    
            list.add(new Student("uranus", 001));
            list.add(new Student("test", 002));
            list.add(new Student("uranus", 001));
    
            ArrayList<Student> list2 = (ArrayList<Student>) ((ArrayList<Student>) list).clone();
    
            System.out.println(list.get(0).getName());
            list2.get(0).setName("uranusleon");
            System.out.println(list.get(0).getName()); // 输出 uranusleon
        }
    }
    
    class Student
    {
        private String name;
        private int stuNo;
    
        public Student(String name, int stuNo)
        {
            this.name = name;
            this.stuNo = stuNo;
        }
    
        public void setName(String name)
        {
            this.name = name;
        }
    
        public String getName() {
            return this.name;
        }
    }
    

    ArrayList本质上是一个数组在维护,数组的元素都是引用。list2拷贝list后数组中的元素和list中的相同,所以对list2数组中引用的对象的修改会引起list的变化。比如将list2中第一个元素对应的对象的名字改为"uranusleon",list.get(0).getName()会输出"uranusleon".

  • 深拷贝

ArrayList中的Iterator

  1. ArrayList是通过内部私有类Itr实现Iterator

  2. 定义的类变量

    int cursor;       // index of next element to return
    int lastRet = -1; // index of last element returned; -1 if no suc
    
  3. Itr.hasNext()判断列表是否有下一个元素

    public boolean hasNext() {
        return cursor != size; //比较cursor和size,因为数组是从0开始计数的,如果相同则说明列表没有下一个元素;
    }
    
  4. Itr.next()返回下一个元素

    public E next() {
        checkForComodification();
        int i = cursor;
        if (i >= size)
            throw new NoSuchElementException();
        Object[] elementData = ArrayList.this.elementData;
        if (i >= elementData.length)
            throw new ConcurrentModificationException();
        cursor = i + 1; //
        return (E) elementData[lastRet = i]; //返回cursor对应的元素,lastRet加一;
    }
    
  5. Itr.remove()

    public void remove() {
        if (lastRet < 0)
            throw new IllegalStateException();
        checkForComodification();
    
        try {
            ArrayList.this.remove(lastRet);
            cursor = lastRet;
            lastRet = -1;
            expectedModCount = modCount;
        } catch (IndexOutOfBoundsException ex) {
            throw new ConcurrentModificationException();
        }
    }
    

    ArrayList可以利用Itr.remove()在遍历的时候将元素全部删除。

    public static void main(String[] args)
    {
        List<Student> list = new ArrayList<>();
    
        list.add(new Student("uranus", 001));
        list.add(new Student("test", 002));
        list.add(new Student("uranus", 001));
    
        Iterator iter = list.iterator();
        while(iter.hasNext())
        {
            iter.next();
            iter.remove();
        }
    }
    

Map

Map集合与Set集合、List集合的关系

1.与Set集合的关系
如果 把Map里的所有key放在一起看,它们就组成了一个Set集合(所有的key没有顺序,key与key之间不能重复),实际上Map确实包含了一个keySet()方法,用户返回Map里所有key组成的Set集合。
2.与List集合的关系
如果把Map里的所有value放在一起来看,它们又非常类似于一个List:元素与元素之间可以重复,每个元素可以根据索引来查找,只是Map中索引不再使用整数值,而是以另外一个对象作为索引。

作者:Ruheng

链接:https://www.jianshu.com/p/589d58033841

來源:简书

简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容