依赖注入(DI)和lazy属性是现代软件开发中常见的概念,尤其是在使用框架如Spring、Angular或.NET时。实现支持依赖注入的lazy属性需要结合DI容器的功能和语言特性来处理延迟初始化。下面以SpringBoot(Java)和Angular(TypeScript)为例,说明如何实现支持依赖注入的lazy属性。一、SpringBoot中的@Lazy与依赖注入在Spring中,@Lazy注
时间:2026-03-13 09:46:35 269浏览 收藏
本文深入剖析了 Kotlin 的 `lazy` 委托为何无法直接满足依赖注入场景的核心需求——它仅在首次访问时初始化且不支持运行时参数,而依赖注入要求每次获取都基于最新、有效的容器上下文;文章给出了三种实用解决方案:自定义 `InjectableLazy` 委托实现真正的“按需获取”、在确保上下文已就绪前提下安全使用 `by lazy`、以及更推荐的 Spring 原生 `@Lazy` 注解(用于控制 bean 实例化时机);同时警示开发者避免混淆不同层级的“懒加载”语义,尤其在多上下文、测试替换或非 Spring 管理对象中,精准把控获取时机与上下文生命周期才是避免空指针和过期引用的关键。

为什么 lazy 不能直接用在依赖注入场景
因为 lazy 初始化只执行一次,且不接受参数,而依赖注入通常需要运行时传入容器或上下文(比如 getBean(Class)。直接写 val service by lazy { ctx.getBean 会把 ctx 捕获为闭包变量——如果 ctx 尚未初始化或后续被替换,这个 lazy 值就会失效或抛 NullPointerException。
用 Lazy + 自定义委托实现延迟获取
核心是把“获取逻辑”推迟到每次访问,而不是仅首次。Kotlin 的 Lazy 接口本身支持 getValue,我们可以封装一个委托,让它每次调用都查容器:
class InjectableLazy<T : Any>(private val getter: () -> T) : ReadOnlyProperty<Any?, T> {
override fun getValue(thisRef: Any?, property: KProperty<*>): T = getter()
}
// 使用
val service by InjectableLazy { applicationContext.getBean<MyService>() }
这样既保持了语法简洁,又确保每次访问都走最新容器实例。注意:这不是真正“懒”,而是“按需获取”,适合依赖可能动态刷新的场景(如多上下文、测试 mock 替换)。
如果必须真懒且支持注入上下文,用带参的 lazy 工厂
标准 lazy 不支持传参,但你可以把容器作为外部依赖提前持有,再用 lazy 封装其 getBean 调用:
private val ctx: ApplicationContext必须在声明前已初始化(比如构造函数注入或@Autowired字段)- 然后写
val service by lazy { ctx.getBean—— 这才是真懒,且安全() } - 若
ctx是 lateinit,务必确保在第一次访问service前已完成赋值,否则触发UninitializedPropertyAccessException
Spring 中更推荐用 @Lazy 注解而非手动 lazy 委托
Spring 原生的 @Lazy 是作用于 bean 创建时机的,和 Kotlin 的 lazy 完全不同层级。它让 Spring 在首次 getBean 时才实例化目标 bean,适用于单例 bean 之间的循环依赖或启动性能优化:
@Component class MyService @Autowired constructor(@Lazy private val heavyDep: HeavyDependency)
这种写法由 Spring 容器控制生命周期,比手动委托更可靠;手动 lazy 委托只影响属性访问行为,不改变 bean 实例化策略。混用两者容易造成语义混淆,比如以为 @Lazy + by lazy 是双重懒加载,其实它们解决的是完全不同的问题。
真正难处理的是跨上下文、非 Spring 管理对象里的 lazy 属性——这时候必须自己控制获取时机和上下文有效性,稍有不慎就拿到过期引用或空指针。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《依赖注入(DI)和lazy属性是现代软件开发中常见的概念,尤其是在使用框架如Spring、Angular或.NET时。实现支持依赖注入的lazy属性需要结合DI容器的功能和语言特性来处理延迟初始化。下面以SpringBoot(Java)和Angular(TypeScript)为例,说明如何实现支持依赖注入的lazy属性。一、SpringBoot中的@Lazy与依赖注入在Spring中,@Lazy注解用于延迟初始化Bean,通常用于解决循环依赖问题或优化启动性能。示例代码:@ComponentpublicclassMyService{//延迟加载的依赖@Autowired@LazyprivateAnotherServiceanotherService;publicvoiddoSomething(){//只有第一次调用时才会初始化anotherService.doWork();}}说明:@Lazy会告诉Spring在注入该Bean时不要立即实例化它,而是在首次访问时才进行初始化。这种方式适用于需要延迟加载的Bean,尤其在有循环依赖的情况下非常有用。二、Angular中的懒加载服务(LazyService)在Angular中,虽然没有直接的@Lazy注解,但可以通过懒加载模块或使用Injector来》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
387 收藏
-
177 收藏
-
108 收藏
-
328 收藏
-
236 收藏
-
394 收藏
-
478 收藏
-
328 收藏
-
128 收藏
-
338 收藏
-
473 收藏
-
148 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习