【面试官】你说说对HashSet的理解?

  • 面试官:你说说对HashSet的理解?
  • 面试官:LinkedHashSet呢?
  • 面试官:TreeSet和它们比有什么特性?
  • 面试官:那TreeSet要怎么定制排序?

大家好,我是南哥。

一个Java学习与进阶的领路人,相信对你通关面试进入心心念念的公司有所帮助。

文章目录

  1. Set集合
    1. HashSet
    2. LinkedHashSet
    3. TreeSet
    4. TreeSet自定义排序

1. Set集合

1.1 HashSet

面试官:你说说对HashSet的理解?

Set集合区别于其他三大集合的重要特性就是元素具有唯一性,南友们记不住这个特性的话,有个易记的方法。Set集合为什么要叫Set呢?因为Set集合的命名取自于我们小学数学里的集合论(Set Theory),数学集合一个很重要的概念就是每个元素的值都互不相同。

Set集合常见的有实例有:HashSet、LinkedHashSet、TreeSet,南哥先缕一缕HashSet。

// HashSet类源码
public class HashSet<E> extends AbstractSet<E> implements Set<E>, Cloneable, java.io.Serializable {...}

HashSet底层实现其实是基于HashMap,HashMap的特点就是Key具有唯一性,这一点被HashSet利用了起来,每一个HashMap的Key对应的就是HashSet的元素值。来看看官方源码的解释。

此类实现Set接口,由哈希表(实际上是HashMap实例)支持。它不保证集合的迭代顺序;特别是,它不保证顺序随时间保持不变。此类允许null元素。

我们创建一个HashSet对象,实际上底层创建了一个HashMap对象。

    // HashSet构造方法源码
    public HashSet() {
        map = new HashMap<>();
    }

HashSet一共提供了以下常用方法,不得不说HahSet在业务开发中还是用的没那么多的,南哥在框架源码上看HashSet用的就比较多,比如由Java语言实现的zookeeper框架源码。

(1)添加元素

    public boolean add(E e) {
        return map.put(e, PRESENT)==null;
    }

我们看上面add方法的源码,是不是调用了HashMap的put方法呢?而put方法添加的Key是HashSet的值,Val则是一个空的Object对象。PRESENT是这么定义的。

    // Dummy value to associate with an Object in the backing Map
    private static final Object PRESENT = new Object();

(2)判断元素是否存在

    public boolean contains(Object o) {
        return map.containsKey(o);
    }

HashSet的contains方法同样是调用HashMap判断Key是否存在的方法:containsKey

(3)移除元素

    public boolean remove(Object o) {
        return map.remove(o)==PRESENT;
    }

1.2 LinkedHashSet

面试官:LinkedHashSet呢?

接着轮到LinkedHashSet,同为Set集合之一,它和上文的HashSet有什么区别?南哥卖个关子。

源码对LinkedHashSet的解释。

Hash table and linked list implementation of the Set interface, with predictable iteration order. This implementation differs from HashSet in that it maintains a doubly-linked list running through all of its entries. This linked list defines the iteration ordering, which is the order in which elements were inserted into the set (insertion-order).

源码的大概意思就是:Set接口的哈希表和链表实现,具有可预测的迭代顺序。此实现与HashSet的不同之处在于,它维护一个贯穿其所有条目的双向链表。此链表定义迭代顺序,即**元素插入集合的顺序 (**插入顺序)。

底层数据结构是一条双向链表,每个元素通过指针进行相连,也就有了按插入顺序排序的功能。

知道了LinkedHashSet的特性,看看他的构造方法。

    /**
     * 构造一个新的、空的链接哈希集,具有默认初始容量(16)和负载因子(0.75)。
     */
    public LinkedHashSet() {
        super(16, .75f, true);
    }

这个方法向上调用了底层C语言源码实现的LinedHashMap的

剩余60%内容,订阅专栏后可继续查看/也可单篇购买

Java面试突击 文章被收录于专栏

👉以面试官面试的形式,涵盖了你怒怼大厂面试官、拿下大厂面试所需掌握的核心知识、面试重点! 👉相信一定对你顺利通关面试、拿到理想Offer有所帮助! 👉花费大量精力去制作本专栏,创作不易,各位的支持就是我创作的最大动力!

全部评论
m
1 回复 分享
发布于 08-20 23:15 黑龙江

相关推荐

1、接口除了public&nbsp;和&nbsp;abstract&nbsp;?还可以用什么?可以用static修饰接口方法嘛?从&nbsp;Java&nbsp;8&nbsp;开始,接口可以包含默认方法(default)和静态方法(static)。protected&nbsp;:在接口中不能使用&nbsp;protected&nbsp;修饰符。原因:&nbsp;接口的目的是提供公共&nbsp;API,而&nbsp;protected&nbsp;只能被同一包中的类或子类访问,这与接口的设计理念相悖。因此,接口中的方法和字段只能是&nbsp;public&nbsp;或默认(包私有)。private:从&nbsp;Java&nbsp;9&nbsp;开始,你可以在接口中使用&nbsp;private&nbsp;修饰符。用途:&nbsp;private&nbsp;方法可以用于接口内部的辅助方法,主要是为了减少代码重复和提高代码的封装性。这些&nbsp;private&nbsp;方法不能被实现类访问。2、public&nbsp;class&nbsp;Go&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;static&nbsp;void&nbsp;main(String[]&nbsp;args)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Integer&nbsp;i&nbsp;=&nbsp;0;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;test(i);&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;System.out.println(i);&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;static&nbsp;void&nbsp;test(Integer&nbsp;i){&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;i++;&nbsp;&nbsp;&nbsp;&nbsp;}}输出0,因为i是不可变类型,而不是因为缓存机制。原理和String一样3、springboot环境中,在初始化阶段,static修饰的常量通过配置文件进行赋值应该怎么做?使用&nbsp;@PostConstruct&nbsp;和静态代码块虽然静态变量不能直接被&nbsp;Spring&nbsp;注入,但可以通过一个静态方法在类被加载时执行初始化逻辑@Configurationpublic&nbsp;class&nbsp;MyConstants&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;static&nbsp;String&nbsp;MY_CONSTANT;&nbsp;&nbsp;&nbsp;&nbsp;@Value(&amp;quot;${my.constant}&amp;quot;)&nbsp;&nbsp;&nbsp;&nbsp;private&nbsp;String&nbsp;myConstant;&nbsp;&nbsp;&nbsp;&nbsp;@PostConstruct&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;void&nbsp;init()&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;MY_CONSTANT&nbsp;=&nbsp;myConstant;&nbsp;//&nbsp;在初始化阶段赋值给静态变量&nbsp;&nbsp;&nbsp;&nbsp;}}4、布隆过滤器的路由转发怎么做?API&nbsp;网关&nbsp;or&nbsp;服务发现机制(如&nbsp;Consul、Eureka&nbsp;或&nbsp;Zookeeper)来动态发现存在的服务使用&nbsp;API&nbsp;网关作为所有请求的入口,网关可以根据请求类型路由到不同的服务。
查看3道真题和解析
点赞 评论 收藏
分享
4 4 评论
分享
牛客网
牛客企业服务