Java基础知识之容器(一)

Java 容器


前言:在java开发中我们会大量的使用集合,在这里我将总结常见的集合类,每个集合类的优点和缺点,以便我们能更好的使用集合。


关于简单容器分类的结构图

此处输入图片的描述
此处输入图片的描述

以上是容器的分类情况,其中比较常用的就是图中红色圈出的部分.点线框表示接口,实线框表示具体的类。


1.基本概念

  • Collection:一个独立元素的序列,这些元素都服从一条或多条规则,是大多数集合类型的接口。List必须按插入顺序保存元素,而Set不能有重复元素。Queue按照排队的规则来确定对象产生的顺序(通常与他们被插入的顺序一样)

  • Map:一组成对的"键值对"对象,允许你使用键来查找值。映射表允许我们使用另一个对象来查找某个对象,它也被称作"关联数组".


2.List

  • ArrayList:
    (1) 长于随机访问元素,但是在List的中间的插入和移除元素较慢。如果是大量的随机访问,建议使用ArrayList。
    (2)ArrayList就是数组列表,主要用来装载数据,当我们装载的是基本类型的数据int,long,boolean,short,byte...的时候我们只能存储他们对应的包装类,它的主要底层实现是数组Object[] elementData。
    (3)ArrayList不是线程安全的,但是在发生并发行为时,它会尽可能的抛出ConcurrentModificationException,此为fail-fast机制。
    (4)ArrayList提供了三种方式的构造器,可以构造一个默认初始容量为10的空列表、构造一个指定初始容量的空列表以及构造一个包含指定collection的元素的列表,这些元素按照该collection的迭代器返回它们的顺序排列的。ArrayList的构造函数如下:
    /**
     * 构造一个指定初始容量的空列表
     */
    public ArrayList(int initialCapacity) {
        if (initialCapacity > 0) {
            this.elementData = new Object[initialCapacity];
        } else if (initialCapacity == 0) {
            this.elementData = EMPTY_ELEMENTDATA;
        } else {
            throw new IllegalArgumentException("Illegal Capacity: "+
                                               initialCapacity);
        }
    }

    /**
     *构造一个默认初始容量为10的空列表
     */
    public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }

    /**
     *构造一个包含指定collection的元素的列表
     */
    
    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其实就是采用的是数组(默认是长度为10的数组)。所有ArrayList在读取的时候是具有和数组相似的效率,所以导致对List的插入与移除元素时较慢。
  • LinkedList
    (1)LinkedList在List中间插入删除时比ArrayList更高效,随机访问要差一些。
    (2)LinkedList还添加了可以使其用作栈,队列或双端队列的方法。各种Queue以及栈的行为,由LinkedList提供支持。以下是LinkedList实现栈的功能:
import java.util.LinkedList;
public class Stack<T> {
    private LinkedList<T> storage = new LinkedList<>();
    //加入元素
    public void push(T v) {
        storage.addFirst(v);
    }
    //返回栈顶元素
    public T peek() {
        return storage.getFirst();
    }
    //移除并返回栈顶元素
    public T pop() {
        return storage.removeFirst();
    }
    public boolean empty() {
        return storage.isEmpty();
    }
    public String toString() {
        return storage.toString();
    }
}
注意:
(1)add方法:将指定的元素附加到该列表的末尾
文档方法定义:add(E e)---> Appends the specified element to the end of this list.
使用add方法后,再使用removefirst返回的是插入的第一元素。即列表的第一个元素


(2)addFirst方法:在列表的开头插入指定的元素。
文档方法定义:addFirst(E e)--->Inserts the specified element at the beginning of this list
使用addfirst方法后,再使用removefirst返回的是插入的最后一元素。同样也是列表的第一个元素。




3.Set

set不保存重复的元素,Set中最常被使用的是测试归属性,你可以很容易访问到某个对象是否在某个Set中,正因如此,查找就成为了Set中最重要的操作,因此你通常会选择一个HashSet的实现,它专门对快速查找进行了优化。

  • HashSet
    出于速度因素的考虑,HashSet使用了散列,同时所存储的元素是没有顺序的。下面是样码:
public class SetofInteger {
    public static void main(String[] args) {
        Random rand = new Random(47);
        Set<Integer> intset = new HashSet<Integer>();
        for(int i=0;i<10000;i++) {
            intset.add(rand.nextInt(30));
        }
        System.out.println("HashSet输出结果: "+intset);
    }
}
/*output: 
HashSet输出结果: [0, 1, 2, 3, 4, 5, 7, 6, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 26, 24, 28, 25, 27, 29]
*/

  • TreeSet
    TreeSet将元素存储在红-黑树数据结构中,所存储的元素是有顺序的。内部是使用比较器将基本数据类型进行比较后排序的,但是对于用TreeSet存储多个对象时,对象本身应该有自定义比较器来实现排序,不然会报错。下面是一个小程序,跟HashSet那个类似。
public class SetofInteger {
    public static void main(String[] args) {
        Random rand = new Random(47);
        Set<Integer> treeset = new TreeSet<>();
        for(int i=0;i<10000;i++) {
            treeset.add(rand.nextInt(30));
        }
        System.out.println("TreeSet输出结果: "+treeset);
    }
}
/*output:
TreeSet输出结果: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29]
*/

4.Map

Map以键值对的形式保存元素,是一种将对象(非数字)与对象相关联的设计。HashMap设计用来快速访问,而TreeMap保持键始终处于排序状态,所以HashMap更快。LinkedHashMap保持元素插入的顺序,但是也是通过散列提供了快速访问的能力。

  • HashMap
    通过散列用来快速访问
public class MapTest {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("Tom", 123);
        map.put("Tony", 134);
        map.put("Mike", 125);
        System.out.println(map);
        for(String str:map.keySet()) {
            System.out.println(str+" has:"+map.get(str));
        }
    }
}
/*output:
{Tony=134, Mike=125, Tom=123}
Tony has:134
Mike has:125
Tom has:123
*/

5.迭代器

  • 迭代器(Iterator)
    迭代器是一个对象,它的工作是遍历并选择序列中的对象,而客户端程序员不必知道或关心该序列底层的结构。
    (1)使用iterator方法要求容器返回一个Iterator.
    (2)使用hasnext()检查序列中是否还有其他元素。
    (3)使用next()获取序列中的下一个元素。
    (4)使用remove()将迭代器新近返回的元素删除。
  • ListIterator
    ListIterator是一个更加强大的Iterator的子类型,它只能用于各种List类的访问。ListIterator可以双向移动。它还可以产生相对于迭代器在列表中指向当前位置的前一个或后一个索引,并可以用set()方法替换掉它访问过的最后一个元素。
public class LinkIteratorTest {
    public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        for(int i=0;i<5;i++) {
            list.add(i);
        }
        ListIterator<Integer> it = list.listIterator();
        while(it.hasNext()) {
            System.out.println(it.next()+","+it.nextIndex()+","+it.previousIndex());
        }
    }
}


6.Collection和Collections

  • Collection:Collection 是一个集合接口。它提供了对集合对象进行基本操作的通用接口方法。Collection接口在Java 类库中有很多具体的实现。Collection接口的意义是为各种具体的集合提供了最大化的统一操作方式。
  • Collections:Collections是一个包装类,也就是一个工具类,它包含有各种有关集合操作的静态多态方法。此类不能实例化,它服务于Collection集合。例如:Collections.addAll()方法接收一个Collection对象,以及一个数组或是一个用逗号分隔的列表,将元素添加到Collection中。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,874评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,102评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,676评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,911评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,937评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,935评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,860评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,660评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,113评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,363评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,506评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,238评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,861评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,486评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,674评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,513评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,426评论 2 352

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,621评论 18 399
  • 集合类简介 为什么出现集合类?面向对象语言对事物的体现都是以对象的形式,所以为了方便对多个对象的操作,就要对对象进...
    阿敏其人阅读 1,416评论 0 7
  • 3.3 集合 一方面, 面向对象语言对事物的体现都是以对象的形式,为了方便对多个对象的操作,就要对对象进行存储。另...
    闫子扬阅读 726评论 0 1
  • Collection ├List │├LinkedList │├ArrayList │└Vector │└Stac...
    AndyZX阅读 873评论 0 1
  • 1. “我不要做个胖子。” 白塔默默地在心里念叨了几遍,头一晕直接昏头涨脑的从跑步机掉了下来。这回是完了,还没开始...
    巫其格阅读 29,250评论 251 970