5.Java容器-LinkedHashMap

LinkedHashMap是HashMap的子类,结合了散列表和双向链表,保持插入顺序或访问顺序。它提供了5个构造方法,重写了put、get和remove等方法,实现了LRU缓存策略的可能性。LinkedHashMap的遍历不受初始容量影响,可通过accessOrder参数控制顺序。阅读源码能发现其利用多态实现不同的遍历顺序和功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

LinkedHashMap概述

       LinkedHashMap是HashMap的子类,内部使用双向链表进行顺序的维护,内部类Entry为HashMap的Node的子类;

       底层是散列表和双向链表,插入的顺序是有序的(底层链表致使有序),和HashMap一样,允许key和value为null,初始容量和装载因子对LinkedHashMap的性能影响很大;也是非同步的,也可以调用Collections的synchronizedMap方法使之具有线程安全的能力;


底层数据结构

构造函数

       如果accessOrder为true,那么表示将顺序记录为访问顺序,否则为插入顺序,默认为false(即,LinkedHashMap默认使用的是插入顺序);

       而正是这一参数并对removeEldestEntry方法进行覆盖便可以快速实现一个简单的LRU缓存;

public LinkedHashMap(int initialCapacity, float loadFactor, boolean accessOrder) {
    super(initialCapacity, loadFactor);
    this.accessOrder = accessOrder;
}

        LinkedHashMap有5个构造方法:



LinkedHashMap的域

LinkedHashMap重写的方法

        下面的是两个比较重要的;

       这就印证了我们的LinkedHashMap底层确确实实是散列表和双向链表:在构建新节点时,构建的是LinkedHashMap.Entry 不再是Node;

put方法

       put方法就是HashMap的put方法,在HashMap部分也提到了afterNodeAccess和afterNodeInsertion方法其实是空实现,而在LinkedHashMap中对其进行了实现,Linked的特性也是在这里进行了体现;


//LinkedhHashMap对此方法进行了覆盖
Node<K,V> newNode(int hash, K key, V value, Node<K,V> e) {
    LinkedHashMap.Entry<K,V> p = new LinkedHashMap.Entry<K,V>(hash, key, value, e);
    linkNodeLast(p);
    return p;
}
//linkNodeLast方法负责双链表的连接:
private void linkNodeLast(LinkedHashMap.Entry<K,V> p) {
    LinkedHashMap.Entry<K,V> last = tail;
    tail = p;
    if (last == null)
        head = p;
    else {
        p.before = last;
        last.after = p;
    }
}

afterNodeAccess

顾名思义,此方法应该在一次访问之后被调用,那么什么算是一次访问呢?LInkedHashMap对此做出了定义:

1.put/putAll/p

### 如何使用 `JSONObject.parseObject` 将 JSON 字符串(包含 `WriteMapNullValue` 特性)正确转换为 `LinkedHashMap` 为了确保 JSON 字符串能够被正确解析并保留键值对的顺序,同时支持 `SerializerFeature.WriteMapNullValue` 序列化特性,可以通过指定目标类型为 `LinkedHashMap` 并启用 `Feature.OrderedField` 来完成此操作。 以下是具体方法: 通过调用 `JSON.parseObject(content, LinkedHashMap.class, Feature.OrderedField)` 方法,可以将 JSON 字符串解析为一个有序的 `LinkedHashMap` 实例[^3]。该方法不仅保持了字段的原始顺序,还允许处理可能存在的空值情况。 下面是一个完整的代码示例来展示这一过程: ```java import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.Feature; import com.alibaba.fastjson.serializer.SerializerFeature; import java.util.LinkedHashMap; import java.util.Map; public class JsonExample { public static void main(String[] args) { String content = "{\"text1\":null,\"text2\":\"456\",\"text3\":\"789\"}"; // 使用 SerializerFeature.WriteMapNullValue 进行序列化 String serializedContent = JSON.toJSONString(JSON.parseObject(content), SerializerFeature.WriteMapNullValue); // 反序列化为 LinkedHashMap,并保留字段顺序 LinkedHashMap<String, Object> result = JSON.parseObject(serializedContent, LinkedHashMap.class, Feature.OrderedField); System.out.println(result.toString()); } } ``` 上述代码中,`SerializerFeature.WriteMapNullValue` 确保在序列化过程中不会忽略值为 `null` 的字段[^5];而 `Feature.OrderedField` 则保证了解析后的 `LinkedHashMap` 中字段的顺序与原 JSON 字符串一致。 #### 注意事项 - 如果未指定 `Feature.OrderedField` 参数,则最终得到的 `Map` 对象可能是无序的 `HashMap` 或其他类型的映射结构[^1]。 - 当需要严格控制字段顺序时,推荐始终显式声明目标容器类型以及附加功能选项。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值