spring中向一个单例bean中注入非单例bean的方法详解
作者:北漂程序员 发布时间:2022-07-19 13:14:18
目录
前言
错误实例演示
实现ApplicationContextAware接口
lookup method
lookup method签名
总结
前言
看到这个题目相信很多小伙伴都是懵懵的,平时我们的做法大都是下面的操作
@Component
public class People{
@Autowired
private Man man;
}
这里如果Man是单例的,这种写法是没有问题的,但如果Man是原型的,这样是否会存在问题。
错误实例演示
这里有一个原型(生命周期为prototype)的类
package com.example.myDemo.component;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
@Component
@Scope(value = "prototype")
public class Man {
public void eat() {
System.out.println("I like beef");
}
}
有一个单例(生命周期为singleton)的类
package com.example.myDemo.component;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.stereotype.Component;
@Component
public class Woman {
//使用依赖注入的方式,注入原型的Man @Autowired
private Man man;
public void eat() {
System.out.println("man:"+man);
System.out.println("I like fruits");
}
}
下面看测试方法,
package com.example.myDemo;
import com.example.myDemo.component.MyFactoryBean;
import com.example.myDemo.component.Woman;
import com.example.myDemo.po.Student;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration;
import org.springframework.context.ApplicationContext;
@SpringBootApplication(exclude={DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class})
public class MyDemoApplication {
public static void main(String[] args) {
ApplicationContext ac=SpringApplication.run(MyDemoApplication.class, args);
Woman woman=(Woman)ac.getBean("woman");
for(int i=0;i<5;i++){
woman.eat();
}
}
}
看下测试结果,
上面的结果显示Woman中的man是单例的,因为5次循环打印打出的结果是同一个对象,发生了什么,
Woman是单例的,Man是原型的,我们使用常规的@Autowired注解注入的却是同一个实例,这里想下为什么Man是一个对象,Woman是单例的,意味着在整个spring容器中只有一个实例,在属性注入的时候肯定也只会注入一次,所以其中Man属性也只能是一个实例,出现上图的结果也就不稀奇了。
现在有这样一个需求要向单例bean中注入原型bean,要怎么实现这样的需求
实现ApplicationContextAware接口
都知道ApplicationContextAware接口是spring提供的一个扩展点,实现该接口的类可以获得ApplicationContext
Woamn类改成下面的样子
package com.example.myDemo.component;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public class Woman implements ApplicationContextAware {
private Man man;
private ApplicationContext ac;
public void eat() {
this.man = (Man) ac.getBean("man");
System.out.println("man:" + man);
System.out.println("I like fruits");
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.ac = applicationContext;
}
}
Woman实现了ApplicationContextAware接口,注入了ApplicaitonContext对象,然后再eat()方法中通过AppicationContext获得Man的实例,看测试结果,
可以看到man属性是多例的也就是符合原型模式的定义。
思考下为什么采用这种方式可以达到注入原型bean的目的
在eat()方法中使用ApplicationContext的getBean方法获取Man,eat()方法每执行一次均会调用一次getBean方法,getbean方法在执行的时候的时候会判断Man的生命周期,如果是原型(prototype)的,那么每调用一次就会重新实例化一个Man,所以会出现上述的结果。
该方法有一个很大的缺点那就是和spring耦合度太高,不符合降低系统的耦合度的要求。
lookup method
spring也考虑了向一个单例bean中注入原型bean的情况,提供了@Lookup注解,在XML配置方式下是<lookup-method>标签,这里仅使用注解的方式演示,
Woman类修改如下,
package com.example.myDemo.component;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Lookup;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public class Woman {
private Man man;
public void eat() {
this.man = createMan();
System.out.println("man:" + man);
System.out.println("I like fruits");
}
@Lookup
public Man createMan(){
return null;
}
}
看下测试结果,
上图显示man是一个多例的,也就是向单例bean中注入了原型bean,其作用的是@Lookup注解。
通过@Lookup注解便完成了注入原型bean的目的,留个思考问题spring是如何做到的?
lookup method签名
被@Lookup注解或<lookup-method>配置的方法有如下要求,
public|protected [abstract] return-type methodName(no-argments)
方法可以是public也可以是protected;
方法可以是抽象的也可以是非抽象的;
方法的返回值是要注入的类型,这里是prototype类型的类;
方法没有入参;
方法体可以是空的。具体返回值可以是null或任何类型,对结果没有影响;
总结
分享了向单例bean中注入原型bean的方式,使用lookup的方式会更简洁些。
这还可能是道面试题哦,各位小伙伴注意喽。lookup的原理下次分享,敬请关注
来源:https://www.cnblogs.com/teach/p/15026142.html


猜你喜欢
- 在编写ui界面时因为手机分辨率大小不同,所以展现出来的效果也是不同的,这个时候就需要考虑适配器,让根据手机分辨率自动适配相应尺寸来展示界面,
- 今天在编译Java程序时遇到如下问题:No enclosing instance of type PrintListFromTailToHe
- Flutter中的默认导航分成两种,一种是命名的路由,一种是构建路由。一、命名路由传参应用入口处定义路由表class MyApp exten
- 目录一、有效的线程1. 如何使用后台线程以避免前台负载过荷?2.如何避免应用不响应ANR?3. 如何在分离的线程中初始化查询?4.其他二、设
- 简介现在市面上的apk只要涉及用户中心都会有头像,而且这个头像也是可自定义的,有的会采取读取相册选择其中一张作为需求照片,另一种就是调用系统
- 1.登录腾讯云点击登录选择浏览器登录。输入用户名 按回车键 然后输入 密码。2.安装java环境直接命令:yum -y install ja
- 增加了用于处理MyBatis的两个bean:SqlSessionFactoryBean、MapperFactoryBean1、注册SqlSe
- 本文实例为大家分享了java实现图书馆管理系统的具体代码,供大家参考,具体内容如下思路:所有包都在book_manage包里利用面向对象的多
- 面向对象的程序是由对象组成的,每个对象包含对用户公开的特定功能部分和隐藏的实现部分。在面向对象程序设计(OOP)中,不必关心对象的具体实现。
- 读取properties配置文件package com.easycrud.utils;import java.io.IOException;
- 关于Android 6.0Android老版本网络请求:1,HttpUrlConnection2,Apache Http ClientAnd
- 一直以来,Java/Spring开发被认为是笨重的代表,无法快速生成项目原型和骨架。所以,Spring推出了Spring Roo这个项目,帮
- Kotlin简介Kotlin是一种针对Java 平台的新编程语言。Kotlin简洁、安全、务实,并且专注于与Java代码的互操作性。它几乎可
- 本文实例为大家分享TextBox和PasswordBox加水印的方法,供大家参考,具体内容如下Textbox加水印Textbox加水印,需要
- 一、ThreadLocal简介多线程访问同一个共享变量的时候容易出现并发问题,特别是多个线程对一个变量进行写入的时候,为了保证线程安全,一般
- 本文是引用开源图表库框架 MPAndroidChart的LineChart地址:https://github.com/PhilJay/MPA
- 简介switch的新特性可是源远流长,早在JDK 12就以预览功能被引入了,最终在JDK 14成为了正式版本的功能:JEP 361: Swi
- 在很多语音视频软件系统中,经常有将实时的音频或视频录制为文件保存到磁盘的需求,比如,视频监控系统中录制监控到的视频、视频会议系统中录制整个会
- 该篇文章内容较多,包括有rabbitMq相关的一些简单理论介绍,provider消息推送实例,consumer消息消费实例,Direct、T
- 废话不多说,直接上代码,小伙伴们仔细看 * 释吧。/*简单的复制 剪切 粘贴 功能 操作: &nb