【并发编程系列】putIfAbsent和getOrDefault用法

2024-01-19 18:28

本文主要是介绍【并发编程系列】putIfAbsent和getOrDefault用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
img

  • 推荐:kwan 的首页,持续学习,不断总结,共同进步,活到老学到老
  • 导航
    • 檀越剑指大厂系列:全面总结 java 核心技术点,如集合,jvm,并发编程 redis,kafka,Spring,微服务,Netty 等
    • 常用开发工具系列:罗列常用的开发工具,如 IDEA,Mac,Alfred,electerm,Git,typora,apifox 等
    • 数据库系列:详细总结了常用数据库 mysql 技术点,以及工作中遇到的 mysql 问题等
    • 懒人运维系列:总结好用的命令,解放双手不香吗?能用一个命令完成绝不用两个操作
    • 数据结构与算法系列:总结数据结构和算法,不同类型针对性训练,提升编程思维,剑指大厂

非常期待和您一起在这个小小的网络世界里共同探索、学习和成长。💝💝💝 ✨✨ 欢迎订阅本专栏 ✨✨

博客目录

    • 一.putIfAbsent
      • 1.基本介绍
      • 2.putIfAbsent 方法
      • 3.putIfAbsent 的并发安全性
    • 二.getOrDefault
      • 1.getOrDefault 方法
      • 2.示例代码
      • 3.使用场景

一.putIfAbsent

1.基本介绍

在 Java 中,putIfAbsentgetOrDefault 是用于 Map 接口的两个不同的方法,下面分别解释 putIfAbsentgetOrDefault 的用法。

2.putIfAbsent 方法

putIfAbsent 方法用于向映射中添加键值对,但仅在指定的键尚不存在时才添加。如果指定的键已经存在,则不会执行添加操作。该方法常用于确保在并发环境中不会重复添加相同的键值对。以下是示例代码:

import java.util.HashMap;
import java.util.Map;public class PutIfAbsentExample {public static void main(String[] args) {Map<String, Integer> map = new HashMap<>();map.put("key1", 10);map.putIfAbsent("key1", 20); // 不会覆盖现有键值对map.putIfAbsent("key2", 30);System.out.println(map); // 输出: {key1=10, key2=30}}
}

3.putIfAbsent 的并发安全性

putIfAbsent 方法在 Java 中通常用于并发环境中的 ConcurrentMap 实现,例如 ConcurrentHashMap。这个方法的设计是为了在多线程环境下安全地执行添加操作,确保在同一时间只有一个线程能够成功地将键值对添加到映射中。

ConcurrentHashMap 中,putIfAbsent 方法使用了一种称为 “compare-and-swap”(CAS)的机制来确保原子性。CAS 是一种多线程同步的技术,它比传统的锁定机制更轻量级。

putIfAbsent 方法中,首先会尝试用给定的键检索值,然后只有在该键不存在时才尝试添加。整个过程是原子性的,因此在并发情况下,多个线程可以同时调用 putIfAbsent 方法,但只有一个线程最终会成功地将键值对添加到映射中。

以下是一个简单的示例,演示了在并发情况下使用 ConcurrentHashMap 中的 putIfAbsent 方法:

import java.util.concurrent.ConcurrentHashMap;public class ConcurrentExample {public static void main(String[] args) {ConcurrentHashMap<String, Integer> concurrentMap = new ConcurrentHashMap<>();// 线程1尝试添加键值对new Thread(() -> {Integer value = concurrentMap.putIfAbsent("key", 1);System.out.println("Thread 1 - Added: " + value);}).start();// 线程2尝试添加相同的键值对new Thread(() -> {Integer value = concurrentMap.putIfAbsent("key", 2);System.out.println("Thread 2 - Added: " + value);}).start();}
}

在这个例子中,只有一个线程最终会成功地将键值对添加到映射中,而另一个线程会得到 putIfAbsent 方法返回的已存在的值。

putIfAbsent 方法在 ConcurrentMap 中提供了一种并发安全的方式来添加元素,适用于多线程环境。

二.getOrDefault

1.getOrDefault 方法

getOrDefault 方法用于获取指定键的值,如果该键不存在,则返回一个默认值。这个方法对于避免检查键是否存在并处理默认值很有用。

2.示例代码

import java.util.HashMap;
import java.util.Map;public class GetOrDefaultExample {public static void main(String[] args) {Map<String, Integer> map = new HashMap<>();map.put("key1", 10);map.put("key2", 20);int value1 = map.getOrDefault("key1", 0); // 存在的键,返回对应的值int value3 = map.getOrDefault("key3", 0); // 不存在的键,返回默认值 0System.out.println(value1); // 输出: 10System.out.println(value3); // 输出: 0}
}

3.使用场景

getOrDefault 方法在 Java 中的 Map 接口中定义,它用于获取指定键对应的值,如果键不存在,则返回一个默认值。这个方法在许多场景下都非常有用,特别是在处理默认值或避免空指针异常的情况下。以下是一些常用的地方:

  1. 避免空指针异常: 当你使用传统的 get 方法获取值时,如果键不存在,返回的是 null。使用 getOrDefault 可以避免因为空值而引发空指针异常。

    Map<String, Integer> map = new HashMap<>();
    Integer value = map.getOrDefault("key", 0); // 如果键不存在,返回默认值 0
    
  2. 处理默认值: 当你需要获取一个值,如果键不存在则使用一个默认值,而不是 null

    Map<String, String> config = getConfig(); // 从某处获取配置信息的映射
    String username = config.getOrDefault("username", "guest"); // 如果键不存在,返回默认用户名 "guest"
    
  3. 统计元素出现的次数: 在统计元素出现次数时,可以使用 getOrDefault 来为不存在的元素提供一个默认的计数值。

    Map<String, Integer> wordCount = new HashMap<>();
    String word = "apple";// 统计单词出现次数
    wordCount.put(word, wordCount.getOrDefault(word, 0) + 1);
    
  4. 初始化映射: 在初始化映射时,可以使用 getOrDefault 设置默认值。

    Map<String, List<String>> userRoles = new HashMap<>();// 初始化用户角色列表
    userRoles.put("admin", userRoles.getOrDefault("admin", new ArrayList<>()));
    

getOrDefault 是一个很方便的方法,可以简化代码,提高可读性,并且在处理映射中的键值对时提供了灵活性。

觉得有用的话点个赞 👍🏻 呗。
❤️❤️❤️本人水平有限,如有纰漏,欢迎各位大佬评论批评指正!😄😄😄

💘💘💘如果觉得这篇文对你有帮助的话,也请给个点赞、收藏下吧,非常感谢!👍 👍 👍

🔥🔥🔥Stay Hungry Stay Foolish 道阻且长,行则将至,让我们一起加油吧!🌙🌙🌙

img

这篇关于【并发编程系列】putIfAbsent和getOrDefault用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/623283

相关文章

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

MyBatis-Flex BaseMapper的接口基本用法小结

《MyBatis-FlexBaseMapper的接口基本用法小结》本文主要介绍了MyBatis-FlexBaseMapper的接口基本用法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具... 目录MyBATis-Flex简单介绍特性基础方法INSERT① insert② insertSelec

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

深入解析Spring TransactionTemplate 高级用法(示例代码)

《深入解析SpringTransactionTemplate高级用法(示例代码)》TransactionTemplate是Spring框架中一个强大的工具,它允许开发者以编程方式控制事务,通过... 目录1. TransactionTemplate 的核心概念2. 核心接口和类3. TransactionT

数据库使用之union、union all、各种join的用法区别解析

《数据库使用之union、unionall、各种join的用法区别解析》:本文主要介绍SQL中的Union和UnionAll的区别,包括去重与否以及使用时的注意事项,还详细解释了Join关键字,... 目录一、Union 和Union All1、区别:2、注意点:3、具体举例二、Join关键字的区别&php

PyCharm接入DeepSeek实现AI编程的操作流程

《PyCharm接入DeepSeek实现AI编程的操作流程》DeepSeek是一家专注于人工智能技术研发的公司,致力于开发高性能、低成本的AI模型,接下来,我们把DeepSeek接入到PyCharm中... 目录引言效果演示创建API key在PyCharm中下载Continue插件配置Continue引言

oracle中exists和not exists用法举例详解

《oracle中exists和notexists用法举例详解》:本文主要介绍oracle中exists和notexists用法的相关资料,EXISTS用于检测子查询是否返回任何行,而NOTE... 目录基本概念:举例语法pub_name总结 exists (sql 返回结果集为真)not exists (s

Springboot中Jackson用法详解

《Springboot中Jackson用法详解》Springboot自带默认json解析Jackson,可以在不引入其他json解析包情况下,解析json字段,下面我们就来聊聊Springboot中J... 目录前言Jackson用法将对象解析为json字符串将json解析为对象将json文件转换为json

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]