首页 > 编程语言 >Java 中如何实现懒加载

Java 中如何实现懒加载

时间:2024-12-27 16:28:31浏览次数:8  
标签:Lazy Java get value 如何 Supplier supplier 加载

Java 中如何实现懒加载

懒加载是一种常见的优化技术,它可以延迟对象的创建或初始化,直到对象第一次被使用时才进行。这种技术可以帮助我们减少资源的浪费,提高程序的运行效率。

在 Scala 中,我们可以使用关键字 lazy 来定义惰性变量,实现延迟加载(懒加载)。但是在 Java 中,我们需要使用其他的技术来实现懒加载。在本文中,我们将介绍如何使用 Java 中的 Supplier 接口和双重检查锁定模式来实现懒加载,并保证只初始化一次。

使用 Supplier 接口实现懒加载

Java 中的 Supplier 接口是一个函数式接口,用于提供类型为 T 的对象。我们可以通过传递一个 lambda 表达式给 Supplier 接口的实例来实现懒加载。

下面是一个使用 Supplier 接口实现懒加载的示例代码:

javaCopy code
import java.util.function.Supplier;

public class Lazy<T> {
    private final Supplier<T> supplier;
    private T value;

    public Lazy(Supplier<T> supplier) {
        this.supplier = supplier;
    }

    public T get() {
        if (value == null) {
            value = supplier.get();
        }
        return value;
    }
}

在上面的代码中,我们定义了一个泛型类 Lazy<T>,并在构造函数中传入一个 Supplier<T> 对象。在 get() 方法中,我们使用 value 变量来缓存 T 类型的对象,并在需要时调用 supplier.get() 方法获取 T 类型的对象。由于 value 变量只会被初始化一次,因此能够保证只有在需要时才会初始化 value 变量。

以下是使用 Lazy 类的示例代码:

javaCopy code
public class LazyDemo {
    public static void main(String[] args) {
        Lazy<String> lazyString = new Lazy<>(() -> {
            System.out.println("Initializing lazy string...");
            return "Hello, World!";
        });

        System.out.println(lazyString.get());
        System.out.println(lazyString.get());
        System.out.println(lazyString.get());
    }
}

在上面的代码中,我们创建了一个 Lazy<String> 对象,并传入一个 lambda 表达式,用于提供 String 类型的对象。在 main() 方法中,我们多次调用 lazyString.get() 方法,并打印返回值。由于 value 变量只会被初始化一次,因此只有在第一次调用 lazyString.get() 方法时会输出 "Initializing lazy string...",后续调用时不会输出。

使用双重检查锁定模式实现懒加载

双重检查锁定模式是一种常见的用于实现懒加载的技术。它利用了同步块和 volatile 关键字来确保线程安全和懒加载。

以下是使用双重检查锁定模式实现懒加载

第一步,创建一个Java类,并声明一个泛型类型,以存储惰性计算的值。在我们的示例中,我们将使用泛型类型T,以便我们可以使用Lazy类来存储任何类型的值。

javaCopy code
public class Lazy<T> {
    private final Supplier<T> supplier;
    private volatile T result;

    public Lazy(Supplier<T> supplier) {
        this.supplier = supplier;
    }

    public T get() {
        T value = result;
        if (value == null) {
            synchronized (this) {
                value = result;
                if (value == null) {
                    value = supplier.get();
                    result = value;
                }
            }
        }
        return value;
    }
}

在上面的代码中,我们声明了一个私有Supplier类型的成员变量supplier,它将计算值的函数作为参数传递。我们还声明了一个volatile类型的成员变量result,用于存储计算的结果,并确保在多线程环境下正确使用。

第二步,实现惰性加载的逻辑。在我们的Lazy类中,我们实现了一个get()方法,该方法返回计算结果。在get()方法中,我们使用了双重检查锁定机制来确保懒加载的正确性。在第一次调用get()方法时,我们检查result变量是否为空。如果为空,我们使用synchronized代码块来避免多个线程同时计算值。在synchronized代码块中,我们再次检查result变量是否为空,以确保在锁定时另一个线程未计算出值。如果为空,我们调用supplier.get()方法来计算值,并将结果存储在result变量中。在计算完成后,我们将值返回给调用者。

第三步,使用单例模式确保只初始化一次。为了确保只初始化一次,我们将result变量声明为volatile类型,并使用双重检查锁定机制。在计算值的过程中,如果另一个线程已经计算了值,则返回先前计算的结果。

第四步,测试我们的Lazy类是否按预期工作。在测试中,我们将创建一个名为TestLazy的类,并声明一个Lazy类型的变量,然后将一个匿名函数传递给Lazy类的构造函数来计算一个值。我们将使用该变量的值来测试惰性加载和单例模式的正确性。

来计算一个值。我们将使用该变量的值来测试惰性加载和单例模式的正确性。

javaCopy code
public class TestLazy {
    public static void main(String[] args) {
        Lazy<Integer> lazyValue = new Lazy<>(() -> {
            int result = 100 + 200;
            System.out.println("Calculating value...");
            return result;
        });

        System.out.println("Before calling get()");
        // 第一次调用
        int value1 = lazyValue.get();
        System.out.println("After calling get()");
        // 第二次调用
        int value2 = lazyValue.get();
        System.out.println("After calling get() again");

        System.out.println("value1: " + value1);
        System.out.println("value2: " + value2);

        // 判断是否为同一个对象
        System.out.println("Is same instance: " + (lazyValue == lazyValue));
    }
}

运行该测试类后,我们期望看到的输出是:

vbnetCopy code
Before calling get()
Calculating value...
After calling get()
After calling get() again
value1: 300
value2: 300
Is same instance: true

输出表明,第一次调用get()方法时,计算值的函数被调用并计算出值。在第二次调用get()方法时,我们没有看到“Calculating value…”这个输出,这证明了惰性加载的正确性。此外,我们还检查了两次获取到的值是否相等,以及对象是否是同一个实例,这证明了单例模式的正确性。

最后,我们现在已经有了一个实现懒加载的Lazy类,该类使用Supplier接口实现了惰性加载和单例模式,使得我们可以轻松地延迟计算值,同时避免了多次初始化变量的问题。

标签:Lazy,Java,get,value,如何,Supplier,supplier,加载
From: https://blog.csdn.net/weixin_74412978/article/details/144675718

相关文章

  • 「 Java基础-链式调用 」Java开发中如何让你的代码看起来更优雅?试试链式调用?
    一、前言我们日常在写业务代码的时候,经常会遇到一种场景,比如一个对象有很多属性,比如用户对象有很多属性:用户名、用户ID、用户性别、用户居住地址、用户工作类型、用户联系方式等等,当我们要构建一个用户对象的时候,就要不断的去set,get如下代码所示:publicclassUser{......
  • JAVA安装部署流程
    JAVA的安装部署流程可以分为以下几个步骤:一、准备工作确定安装环境:确定操作系统版本(如Windows、Linux、MacOS等)。检查系统是否满足JAVA安装的最低要求。下载JAVA安装包:访问Oracle官网或其他可信的JAVA下载站点。根据操作系统版本下载对应的JAVA安装包,通常是JDK(Java......
  • 战损版JavaAgent方法耗时统计工具实现
    新来的实习生妹妹故意刁难我,说想让我实现一个方法耗时统计工具,不能用切面,这能难倒我嘛,JavaAgent安排上。前言本篇文章将实现一个超绝战损版的基于JavaAgent的方法耗时统计工具。整体内容分为:JavaAgent原理简析;方法耗时统计工具实现;方法耗时工具的Springboot的s......
  • [响应式编程] 如何优雅Exception异常处理
    初识响应式编程的时候,除了从命令式的思维方式转变为函数式的编程方式外,其中有一个很大的不适应的地方就是在面对异常时该怎么处理,尤其是面对检查异常(CheckedException)时更是不知所措。在遇到异常时,我们通用的处理方式就是打日志、降级兜底、重试三板斧,本文通过ProjectReacto......
  • 水仓水位异常识别智慧矿山一体机构建智慧矿山系统:监控硬盘如何影响监控系统的稳定性?
    中国作为全球最大的煤炭生产和消费国,煤矿行业在能源供应中依然占据重要地位。虽然国家逐步推动能源结构转型,发展可再生能源,但煤炭仍然在短期内满足能源需求方面发挥着重要作用。近年来,煤矿安全生产水平有所提高,但仍存在一定的安全隐患。国家对煤矿的安全监管力度加大,企业在防范事......
  • node.js毕设 JavaWeb的美食街摊位管理系统 论文+程序
    本系统(程序+源码+数据库+调试部署+开发环境)带文档lw万字以上,文末可获取源码系统程序文件列表开题报告内容一、选题背景关于美食街摊位管理的研究,现有研究多集中在传统管理模式的局部优化方面,如单独对摊位分配或者租赁合同管理的优化。专门针对将摊位管理员、商家、摊位区......
  • 【java毕设 python毕设 大数据毕设】基于springboot的学生宿舍管理系统的设计与实现
    ✍✍计算机毕设编程指导师**⭐⭐个人介绍:自己非常喜欢研究技术问题!专业做Java、Python、小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。⛽⛽实战项目:有源码或者技术上的问题欢迎在评论区一起讨论交流!⚡⚡Java、Python、小程序、大数据实战项目集⚡⚡文末获取......
  • 如何通过JxBrowser将富文本编辑器嵌入到 Java 桌面应用程序中?
    JxBrowser(试用下载)是一个跨平台的JVM库,它允许您将基于Chromium的Browser控件集成到Compose、Swing、JavaFX、SWT应用程序中,并使用Chromium的数百种功能。所有桌面工具包都提供文本编辑控件,从最基本的选项到更高级的选项。但是富文本编辑呢?是否有允许用户格式化文本......
  • 【实用技能】如何利用Visual Paradigm输出项目中图表或模型的注释
    VisualParadigm 包含设计共享、线框图和数据库设计新特性的企业项目设计工具,致力于助力您的IT项目研发!VisualParadigm中的Doc.Composer,可让您以完全可自定义的方式输出模型和图表的每个细节。在本文中,您将学习如何创建自定义模板来输出项目中图表或模型元素的注释。点击......
  • Java 并发编程:原子类(Atomic Classes)核心技术的深度解析
    Java并发编程:原子类(AtomicClasses)核心技术的深度解析在高并发场景下,线程安全是一个重要的话题。Atomic类通过高效的CAS(Compare-And-Swap)机制,为开发者提供了一种无需锁的线程安全解决方案。本篇文章将系统讲解Java原子类的核心概念、常用成员、使用方法以及实际应用。......