使用Guava轻松创建和管理不可变集合

news2025/1/24 17:53:42

第1章:引言

大家好,我是小黑。今天,我们来聊聊一个在Java编程里超有用的话题:使用Guava创建和管理不可变集合。首先,咱们得明白,什么是不可变集合。简单来说,不可变集合就是一旦创建就不能被修改的集合。

为啥要用不可变集合呢?想象一下,你写了一段代码,把一个集合传给了别的方法。如果那个方法不小心改了你的集合,那岂不是一场灾难?但如果你的集合是不可变的,这种情况就绝对不会发生。不可变集合还有助于编写更加清晰、更容易维护的代码,还能提高程序的性能哦。

第2章:Guava不可变集合简介

Guava是Google推出的一个Java库,里面有一堆好用的工具类,其中就包括了不可变集合。Guava的不可变集合和咱们平时用的Java标准库集合有啥不同呢?主要是Guava的集合一旦创建,就不能被修改,这就大大减少了出错的可能性。

来,让我给你演示一下怎么用Guava创建不可变集合。比如说,咱们要创建一个不可变的列表:

import com.google.common.collect.ImmutableList;

public class ImmutableDemo {
    public static void main(String[] args) {
        ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry");
        System.out.println(immutableList);
    }
}

这段代码创建了一个包含三种水果名称的不可变列表。你看,使用ImmutableList.of就能轻松创建。这样一来,无论谁拿到这个列表,都不能添加、删除或者修改里面的元素。

同理,不可变的集合(Set)和映射(Map)也可以用类似的方式创建。只要记住,一旦创建,就不能更改了。这样的特性在很多场景下都非常实用,比如在多线程环境下,不可变集合可以保证数据的安全性,防止出现并发修改异常。

第3章:创建不可变集合

让我们来看看不可变列表(ImmutableList)的创建。咱们已经看过最基本的创建方式了,现在再来看点高级的:

import com.google.common.collect.ImmutableList;

public class ImmutableListDemo {
    public static void main(String[] args) {
        // 创建一个不可变列表
        ImmutableList<String> immutableList = ImmutableList.<String>builder()
                .add("Apple")
                .add("Banana")
                .add("Cherry")
                .build();
        System.out.println(immutableList);
    }
}

在这个例子中,小黑用了ImmutableList.builder()方法。这个方法特别棒,因为它可以让咱们一步步地添加元素,最后再用build()方法一次性创建不可变列表。

接下来,看看不可变集(ImmutableSet)的创建:

import com.google.common.collect.ImmutableSet;

public class ImmutableSetDemo {
    public static void main(String[] args) {
        // 创建一个不可变集
        ImmutableSet<String> immutableSet = ImmutableSet.of("Apple", "Banana", "Cherry");
        System.out.println(immutableSet);
    }
}

ImmutableSet.of()方法和列表的创建很像,也是一次性把所有元素传进去。

我们来看看不可变映射(ImmutableMap)的创建:

import com.google.common.collect.ImmutableMap;

public class ImmutableMapDemo {
    public static void main(String[] args) {
        // 创建一个不可变映射
        ImmutableMap<String, Integer> immutableMap = ImmutableMap.of("Apple", 1, "Banana", 2, "Cherry", 3);
        System.out.println(immutableMap);
    }
}

在这个例子中,ImmutableMap.of()方法是用键值对的方式来创建映射的。这里,“Apple”是键,1是它的值,以此类推。

第4章:不可变集合的优势

安全性

首先是安全性。不可变对象是线程安全的,这意味着在多线程环境中,你完全不需要担心并发修改的问题。因为数据一旦创建就不会改变,所以不会出现线程间的冲突。这在编写并发程序时特别有用。

举个例子吧,假设有这样一个场景:

public class ThreadSafetyDemo {
    public static void main(String[] args) {
        ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry");

        // 多线程环境下访问不可变列表
        Runnable task = () -> {
            for (String fruit : immutableList) {
                System.out.println(Thread.currentThread().getName() + " - " + fruit);
            }
        };

        new Thread(task).start();
        new Thread(task).start();
    }
}

在这个例子中,即使多个线程同时访问immutableList,也不会引发任何问题,因为它是不可变的。

效率

接下来谈谈效率。不可变对象由于状态不变,可以减少内存的占用,因为它们可以被自由地共享。这就意味着在一定条件下,不可变集合比可变集合更加内存高效。

可读性

不可变集合还能提高代码的可读性和维护性。当你看到一个集合是不可变的,你就可以立即确定这个集合在整个生命周期内都不会改变。这就减少了理解和维护代码的复杂性。

使用Guava的不可变集合可以带来很多好处,特别是在处理安全性、效率和可读性方面。虽然它们在某些情况下可能有些限制,但在正确的场景下使用,绝对可以帮助你写出更优雅、更稳定的Java代码。

第5章:使用场景与最佳实践

使用场景
  1. 常量集合:当你想定义一些永远不变的数据,比如配置项、选项列表等,不可变集合是最佳选择。它们可以公开访问而不用担心被意外修改。

    public class Constants {
        public static final ImmutableList<String> FRUITS = ImmutableList.of("Apple", "Banana", "Cherry");
        // 其他常量...
    }
    
  2. 类内部状态:在创建类时,使用不可变集合来保存内部状态可以确保类的实例一旦创建就保持不变,这对于创建不可变类(immutable class)非常有用。

    public class UserPreferences {
        private final ImmutableSet<String> preferences;
    
        public UserPreferences(Set<String> preferences) {
            this.preferences = ImmutableSet.copyOf(preferences);
        }
    
        // Getter方法...
    }
    
  3. 函数返回值:当函数需要返回集合时,使用不可变集合作为返回类型可以确保调用者不会修改这个集合,从而保证了数据的完整性。

    public class ProductService {
        public ImmutableList<Product> getAvailableProducts() {
            // 查询并返回产品列表
        }
    }
    
最佳实践
  1. 避免预期外的修改:使用不可变集合可以防止调用者意外修改集合内容,从而导致难以追踪的bug。

  2. 提前拷贝:当从可变集合创建不可变集合时,应当在创建时就进行拷贝,以确保不可变集合的独立性。

    public class ConfigLoader {
        public ImmutableSet<String> loadConfig(Set<String> mutableSet) {
            return ImmutableSet.copyOf(mutableSet);
        }
    }
    
  3. 在构造函数中使用不可变集合:当创建对象时,可以使用Guava的不可变集合作为构造函数参数,这样可以在对象创建时就确保其不可变性。

    public class Employee {
        private final ImmutableList<String> skills;
    
        public Employee(List<String> skills) {
            this.skills = ImmutableList.copyOf(skills);
        }
    
        // Getter方法...
    }
    

第6章:与Java 8及以后版本的整合

利用Streams处理不可变集合

Java 8的Streams API可以和Guava的不可变集合无缝合作。比如说,你可以轻松地将一个不可变集合转换成Stream,进行各种操作,然后再收集回不可变集合。

import com.google.common.collect.ImmutableList;
import java.util.stream.Collectors;

public class StreamIntegrationDemo {
    public static void main(String[] args) {
        ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry");

        // 使用Stream API处理Guava不可变集合
        ImmutableList<String> processedList = immutableList.stream()
                .filter(s -> s.startsWith("B"))
                .collect(ImmutableList.toImmutableList());

        System.out.println(processedList);
    }
}

在这个例子里,小黑先把一个不可变列表转换成了Stream,然后用filter方法筛选出以"B"开头的元素,最后再收集回一个不可变列表。这就是Java 8 Stream和Guava不可变集合强强联合的例子。

使用Lambda表达式

Java 8的Lambda表达式也可以和Guava的不可变集合很好地结合。它们可以使得对集合的操作更加简洁。

import com.google.common.collect.ImmutableList;

public class LambdaIntegrationDemo {
    public static void main(String[] args) {
        ImmutableList<String> immutableList = ImmutableList.of("Apple", "Banana", "Cherry");

        // 使用Lambda表达式遍历不可变集合
        immutableList.forEach(fruit -> System.out.println("Fruit: " + fruit));
    }
}

在这个例子中,咱们使用了forEach方法和Lambda表达式来遍历不可变集合。这种方式比传统的for循环更简洁,更易读。

Guava的不可变集合与Java 8及以后版本的特性相结合,可以提供更强大的数据处理能力,同时让代码变得更加简洁和易于理解。

第7章:避免常见陷阱

不可变集合不等于只读集合

首先要清楚,Guava的不可变集合和只读集合不是一回事。不可变集合是在创建时就确定了内容,而只读集合只是不能修改,其底层数据可能被其他引用修改。看个例子:

import com.google.common.collect.ImmutableList;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

public class ImmutableVsReadOnly {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("Apple");
        list.add("Banana");

        // 创建只读集合
        List<String> readOnlyList = Collections.unmodifiableList(list);

        // 创建不可变集合
        ImmutableList<String> immutableList = ImmutableList.copyOf(list);

        // 修改原始列表
        list.add("Cherry");

        // 只读集合的内容发生了变化
        System.out.println("Read-only list: " + readOnlyList);

        // 不可变集合的内容没变
        System.out.println("Immutable list: " + immutableList);
    }
}

这个例子展示了只读集合和不可变集合的区别。当原始集合修改时,只读集合的内容也会跟着变,但不可变集合的内容不会变。

注意构建时的副作用

在使用ImmutableList.builder()或类似的构建器时,要注意不要引入副作用。比如,不要在添加元素的过程中修改这些元素。

// 错误示范:在构建不可变集合时修改元素
ImmutableList.Builder<String> builder = ImmutableList.builder();
for (String fruit : fruits) {
    modifyFruit(fruit); // 不应在这里修改fruit
    builder.add(fruit);
}
小心空指针异常

Guava的不可变集合在创建时会对元素进行非空校验,这意味着如果你试图添加一个null元素,它会立即抛出NullPointerException

// 这将抛出NullPointerException
ImmutableList<String> list = ImmutableList.of("Apple", null, "Cherry");

第8章:总结

选择不可变集合不仅仅是为了编码的方便,更重要的是它们提供了额外的安全性、效率和可维护性。在多线程环境下,不可变集合几乎是必不可少的,因为它们天生就是线程安全的。此外,它们还能帮助减少bug和意外行为,尤其是在大型和复杂的项目中。

要注意,在某些情况下,使用不可变集合可能会有性能开销,特别是在需要频繁修改集合的场景中。因此,选择合适的数据结构和集合类型对于任何项目都至关重要。

希望这篇博客能够帮助你更好地理解Guava的不可变集合,让你在Java编程的道路上更进一步。如果你有任何问题或者想深入讨论,欢迎在评论区留言,小黑很乐意和你一起探讨和学习!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1329979.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

深入了解 Android 中的应用程序签名

深入了解 Android 中的应用程序签名 一、应用程序签名介绍1.1 应用程序签名1.2 应用程序签名的意义1.3 应用程序签名的流程1.4 应用程序签名的方案1.5 签名的重要性和应用场景 二、AOSP 的应用签名2.1 AOSP的应用签名文件路径2.2 应用程序指定签名文件 三、Android Studio 的应…

【SpringBoot篇】基于Redis实现生成全局唯一ID的方法

文章目录 &#x1f354;生成全局唯一ID&#x1f339;为什么要生成全局唯一id&#x1f33a;生成全局id的方法✨代码实现 &#x1f354;生成全局唯一ID 是一种在分布式系统下用来生成全局唯一id的工具 在项目中生成全局唯一ID有很多好处&#xff0c;其中包括&#xff1a; 数据…

在Next.js和React中搭建Cesium项目

在Next.js和React中搭建Cesium项目&#xff0c;需要确保Cesium能够与服务端渲染(SSR)兼容&#xff0c;因为Next.js默认是SSR的。Cesium是一个基于WebGL的地理信息可视化库&#xff0c;通常用于在网页中展示三维地球或地图。下面是一个基本的步骤&#xff0c;用于在Next.js项目中…

信号与线性系统翻转课堂笔记8——周期信号的频谱

信号与线性系统翻转课堂笔记8——周期性信号的频谱 The Flipped Classroom8 of Signals and Linear Systems 对应教材&#xff1a;《信号与线性系统分析&#xff08;第五版&#xff09;》高等教育出版社&#xff0c;吴大正著 一、要点 &#xff08;1&#xff0c;重点&#…

【github】github设置项目为私有

点击setting change to private 无脑下一步

【Proteus/8086】swjtu西南交大微机与接口技术实验:计时器

实验内容: 计时器基本功能: 1)CPU必须用8086 2)计时器最小计时单位为秒。 3)以00:00格式显示计时,前2位表示分钟,后2位表示秒。 4)计时器是正计时方式 5)有暂停、继续计时功能 6&#xff09;有复位计时功能 7&#xff09;每次按下暂停键&#xff0c;能显示计时间隔时间 参考…

uniapp中使用封装步骤条组件

针对步骤条封装完终于清清楚楚啦 先看效果&#xff1a; 附上代码&#xff1a;使用可直接复用&#xff1a;数据是写在了当前组件中&#xff0c;如有必须&#xff0c;可以使用其中的props传值stepInfos传递相应的数据&#xff0c;根据steps步数就可以控制走到哪一步啦 <temp…

双向数据绑定是什么

一、什么是双向绑定 我们先从单向绑定切入单向绑定非常简单&#xff0c;就是把Model绑定到View&#xff0c;当我们用JavaScript代码更新Model时&#xff0c;View就会自动更新双向绑定就很容易联想到了&#xff0c;在单向绑定的基础上&#xff0c;用户更新了View&#xff0c;Mo…

在 Windows 中关闭指定端口的方法

方法一&#xff1a;使用命令行&#xff08;Command Prompt&#xff09; 查找端口占用情况 打开命令提示符&#xff08;Command Prompt&#xff09;并输入以下命令来查找占用指定端口的进程&#xff1a; netstat -aon|findstr "<port_number>" 这里的 <p…

【笔记】Spring的循环依赖

Spring的循环依赖 ObjectFactory:函数式接口&#xff0c;可以将lambda表达式作为参数放在方法的实参种&#xff0c;在方法执行的时候&#xff0c;并不会实际的调用当前lambda表达式&#xff0c;只有在调用getObject方法的时候才回去调用lambda表达式 为什么spring要用三级缓存…

一周工作问题总结(2023.12.18-2023.12.22)

一周工作问题总结 1. 接口调用频率2. 汉字在数据库中占用字节问题3. Map在循环中修改自己的key与value4. Group BY5.递归同步数据6.代码移动Idea飘红 1. 接口调用频率 供应商给的接口可以每秒调用5-10次&#xff0c;那么我为了保险每秒调用5次&#xff0c;为了防止过度调用接口…

SpringBoot+SSM项目实战 苍穹外卖(7)(Spring Cache)

继续上一节的内容&#xff0c;本节实现缓存菜品、缓存套餐、添加购物车、查看购物车和清空购物车功能。 目录 缓存菜品缓存套餐(基于Spring Cache)EnableCaching、Cacheable、CachePut和CacheEvictSpring Cache实现缓存套餐 添加购物车查看购物车清空购物车 缓存菜品 用户端小…

nodejs+vue+ElementUi医院预约挂号系统3e3g0

本医院预约挂号系统有管理员&#xff0c;医生和用户。该系统将采用B/S结构模式&#xff0c;使用Vue和ElementUI框架搭建前端页面&#xff0c;后端使用Nodejs来搭建服务器&#xff0c;并使用MySQL&#xff0c;通过axios完成前后端的交互 管理员功能有个人中心&#xff0c;用户管…

使用 Elasticsearch 检测抄袭 (一)

作者&#xff1a;Priscilla Parodi 抄袭可以是直接的&#xff0c;涉及复制部分或全部内容&#xff0c;也可以是释义的&#xff0c;即通过更改一些单词或短语来重新表述作者的作品。 灵感和释义之间是有区别的。 即使你得出类似的结论&#xff0c;也可以阅读内容&#xff0c;获得…

java练习题之多态练习

1&#xff1a;关于多态描述错误的是(D) A. 父类型的引用指向不同的子类对象 B. 用引用调用方法&#xff0c;只能调用引用中声明的方法 C. 如果子类覆盖了父类中方法&#xff0c;则调用子类覆盖后的方法 D. 子类对象类型会随着引用类型的改变而改变 2&#xff1a;class Supe…

【ONE·MySQL || 基础介绍】

总言 主要内容&#xff1a;MySQL在Centos 7下的安装&#xff08;主要学习相关指令语句&#xff0c;理解安装操作是在做什么&#xff09;、对MySQL数据库有一个基础认识。 文章目录 总言0、MySQL的安装与卸载&#xff08;Centos 7&#xff09;0.1、MySQL的卸载0.1.1、卸载不必要…

C练习题13答案

单项选择题(本大题共20小题,每小题2分,共40分。在每小题给出的四个备选项中,选出一个正确的答案,并将所选项前的字母填写在答题纸的相应位置上。) 1.结构化程序由三种基本结构组成、三种基本结构组成的算法是(A) A.可以完成任何复杂的任务 B. 只能完成部分复杂的任务 C. 只能完…

没通知就降公积金的脉脉,面试考什么?

写在前面 刚过去的周末&#xff0c;又一家神奇公司映入眼帘&#xff1a; 怎么说呢&#xff1f;自家的瓜出现在自家&#xff0c;至少说明脉脉没有跟微博似的限流。 自家做职场社交&#xff0c;就用自家帖子做全员通知&#xff0c;脉脉你就这么“降本”的吗&#xff1f;&#x1f…

【OCR识别】PaddleHub实现验证码识别

文章目录 前言无脑安装使用PaddleHub寻找预训练模型库完整代码效果图 前言 前面有篇文章介绍了 【网站验证码识别】 &#xff0c;但是其是利用 tesseract 工具的命令行来实现图片内容的识别。 这几天我突然想起&#xff0c;大学时参加百度 AI 比赛用过其 PaddleHub 框架&…

计算机网络 应用层上 | 域名解析系统DNS 文件传输协议FTP,NFS 万维网URL HTTP HTML

文章目录 1 域名系统DNS1.1 域名vsIP&#xff1f;1.2 域名结构1.3 域名到IP的解析过程域名服务器类型 2 文件传送协议2.1 FTP 文件传输协议2.2 NFS 协议2.3 简单文件传送协议 TFTP 3 万维网WWW3.1 统一资源定位符URL3.2 超文本传送协议HTTP3.2.1 HTTP工作流程3.2.2 HTTP报文结构…