校招面试之Java容器

最近校招季,特把自己面试中遇到的问题整理整理,以巩固自己的知识。

Java中对于容器有两大类存储方式,一种是单元素存放,还有一种就是key-value这种有关联的双元素存放了。对于Java中的容器,有下列的结构图可以参照:

Collection (用来存放独立元素的序列)
├List 
│├LinkedList 
│├ArrayList 
│└Vector 
│ └Stack 
└Set 
 ├HashSet
 └TreeSet
Map (用来存放key-value型的元素对)
├Hashtable 
├HashMap 
├TreeMap 
└WeakHashMap

下面,我们就来分别讲讲这几种容器。

List

List是有序的Collection,使用此接口能控制每个元素插入的位置,用户能够使用索引来访问元素。与Set不同的是,List允许有重复的元素在其中。

  • ArrayList
    ArrayList相当于顺式存储(线性表),当实例化一个ArrayList时,一个数组也被实例化了,默认初始化一个长度为10的数组。当添加数据的时候会判断当前容量是否能够容下新增的对象,一旦发现容量不足,会自动扩容,新的大小为原有容量的1.5倍+1。

    ArrayList可以快速随机访问,通过调用get(i)方法来访问下标为i的元素。

  • LindedList
    LinkedList相当于链式存储(双向链表),它是通过节点直接彼此连接来实现的。每一个节点都包括前一个节点的引用,后一个节点的引用和节点存储的值。
    当插入或删除节点时,只需要修改其中保持先后关系的节点的引用即可,所以,操作其中的对象速度比ArrayList要快的多。

    但是LinkedList不能随机访问元素,虽然它有get()方法,但是这个方法是通过遍历节点来定位的,速度很慢。

  • Vector

    Vector和ArrayList一样,也是用数组来存储元素。但是Vector使用了synchronized方法,线程安全,所以在性能上比ArrayList要差。

ArrayList和LinkedList的区别

  • ArrayList实现了基于动态数组的数据结构,LinkedList实现了基于链表的数据结构
  • 对于随机访问get和set,ArrayList优于LinkedList
  • 对于增删add和remove,LinkedList优于ArrayList

Set

Set是一种不包含重复元素的Collection,它的构造函数有一个约束条件,传入的Collection参数不能包含重复的元素。

  • HashSet
    HashSet实现了Set接口,由哈希表支持。它不保证Set的迭代顺序,特别是它不保证该顺序恒久不变。

    HashSet底层使用的容器实际上就是HashMap,它以HashMap的key来保存所有的元素,value使用一个static final的Object对象来标识。

    private static final Object PRESENT = new Object();
    
  • TreeSet
    TreeSet整体上性能没有HashSet好,但是它可以维持元素的排序状态。

    TreeSet底层使用的容器实际上就是TreeMap,它以TreeMap的key来保存set集合的元素,value都以一个名为PRESENT的Object对象代替(无实际意义)。

Map

Map接口提供key到value的映射,一个Map不能包含相同的key,每个key只能映射一个value。

  • HashMap
    HashMap底层就是一个数组结构,数组中的每一项又是一个链表(其实就是哈希表的拉链法实现)。但是此类不保证映射的顺序,特别是不保证该顺序恒久不变。但是TreeMap可以维持映射的顺序。

  • Hashtable
    和HashMap实现差不多,具体区别见下面的Hashtable和HashMap的区别

  • TreeMap
    TreeMap的底层实现是一个红黑树结构,这样可以保证快速检索节点,TreeMap可以维持映射的顺序。

    下面我们来具体说下TreeMap的底层实现,首先我们需要了解下排序二叉树:

    • 排序二叉树:要么是一棵空二叉树,要么是具有下列性质的二叉树:
      1. 若它的左子树不为空,则左子树上所有节点的值均小于根节点的值
      2. 若它的右子树不为空,则右子树上所有节点的值均大于根节点的值
      3. 它的左右自子树也分别为排序二叉树

    对于排序二叉树,它的中序遍历就可以得到由小到大的有序序列,所以用它就可以实现快速检索,但是为什么Java还要多此一举用红黑树呢?

    • 排序二叉树虽然可以快速检索,但是在最坏情况下:若插入的节点本身就是有序的,要么由小到大排列,要么由大到小排列,那么最后得到的排序二叉树就变为了链表:所有的节点只有左节点或者所有的节点只有右节点。这种情况下,排序二叉树就变为了普通链表,检索效率会很差。

    所以,Java引入了红黑树作为TreeMap的底层实现

    • 红黑树:红黑树是一种更高效的检索二叉树,它的性质为:

      1. 所有的节点都为红色或者黑色
      2. 根节点永远是黑色
      3. 所有的叶节点都是空节点(NULL),并且是黑色
      4. 每个红色节点的两个子节点都是黑色,即从根节点到叶子节点的路径上不会出现两个连续的红色节点。
      5. 从任一节点到其子树中每个叶子节点的路径都包含相同数量的黑色节点

      红黑树通过上面的限制来保证它大致是平衡的(因为红黑树的高度不会无限增高),这样保证了红黑树在最坏情况下都是高效的,不会出现普通排序二叉树的情况。

Hashtable和HashMap的区别

  • 继承和实现不同
    Hashtable是继承自陈旧的Dictionary类,实现了Map接口;HashMap实现接口(继承自AbstractMap,AbstractMap实现Map接口)

  • 线程安全不同
    Hashtable是线程安全的,它的实现方法里面都添加了synchronized关键字来确保线程同步。HashMap是线程不安全的,在多线程编程下如使用HashMap需要使用Collections.synchronizedMap()来获取一个线程安全的集合。

  • 对null的处理不同
    HashMap支持null作为key和value,但是Hashtable不允许(key,value都不允许)。HashMap的方法get()返回null时,既可以表示没有改键,也可以表示该键对应的值为null,所以不能用此判断是否有该键,而应该用containsKey()。

  • HashMap初始容量为16,Hashtable初始容量为11。HashMap扩容时是当前容量翻倍:capacity2,Hashtable是当前容量翻倍+1:capacity2+1。

  • 哈希值的使用不同
    Hashtable直接使用key的hashcode对table数组的长度取模,HashMap是对key的hashcode进行二次hash,然后对table数组的长度取模,以获得更好的散列值。

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

推荐阅读更多精彩内容