zoukankan      html  css  js  c++  java
  • spring中@Lookup的作用

    情景分析

      在Spring的诸多应用场景中bean都是单例形式,当一个单利bean需要和一个非单利bean组合使用或者一个非单利bean和另一个非单利bean组合使用时,我们通常都是将依赖以属性的方式放到bean中来引用,然后以@Autowired来标记需要注入的属性。但是这种方式在bean的生命周期不同时将会出现很明显的问题,假设单利bean A需要一个非单利bean B(原型),我们在A中注入bean B,每次调用bean A中的方法时都会用到bean B,我们知道Spring Ioc容器只在容器初始化时执行一次,也就是bean A中的依赖bean B只有一次注入的机会,但是实际上bean B我们需要的是每次调用方法时都获取一个新的对象(原型)所以问题明显就是:我们需要bean B是一个原型bean,而事实上bean B的依赖只注入了一次变成了事实上的单利bean。

      

    代码示例:

    @Component
    @Scope("prototype")
    public class PrototypeBean {
        private static final Logger logger= LoggerFactory.getLogger(PrototypeBean.class);
        
        public void say() {
            logger.info("say something...");
        }
    }
    @Component
    public class SingletonBean {
        private static final Logger logger = LoggerFactory.getLogger(SingletonBean.class);
        
        @Autowired
        private PrototypeBean bean;
        
        public void print() {
            logger.info("Bean SingletonBean's HashCode : {}",bean.hashCode());
            bean.say();
        }
    }
    @SpringBootApplication
    public class SampleApplication {
        private static final Logger logger = LoggerFactory.getLogger(SampleApplication.class);
        public static void main(String[] args) {
            SpringApplication.run(SampleApplication.class, args);
        }
    
        @Bean 
    public CommandLineRunner test(final SingletonBean bean) { return (args)-> { logger.info("测试单例bean和原型bean的调用"); int i =0; while(i<3) { i++; bean.print(); } }; } }

    结果

    2018-12-06 15:04:29,721 INFO :-- [main .. ] o.s.SampleApplication 测试单例bean和原型bean的调用 
    2018-12-06 15:04:29,723 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1713129148 
    2018-12-06 15:04:29,723 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:04:29,723 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1713129148 
    2018-12-06 15:04:29,724 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:04:29,724 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1713129148 
    2018-12-06 15:04:29,724 INFO :-- [main .. ] o.s.a.PrototypeBean say something...
    

      我们看到每次输出PrototypeBean的HashCode都是一样的,证明我们实际上并没有达到使用原型bean的目的。

    解决方案

    1. 在bean A中引入ApplicationContext每次调用方法时用上下文的getBean(name,class)方法去重新获取bean B的实例。
    2. 使用@Lookup注解。
      这两种解决方案都能解决我们遇到的问题,但是第二种相对而言更简单。以下给出两种解决方案的代码示例。

       通过应用上下文ApplicationContext获取获取

        

    @Component
    public class SingletonBean {
        private static final Logger logger = LoggerFactory.getLogger(SingletonBean.class);
        
        @Autowired
        private ApplicationContext context;
        
        public void print() {
            //每次都从ApplicatonContext中获取新的bean引用
            PrototypeBean bean = this.context.getBean("prototypeBean",PrototypeBean.class);
            logger.info("Bean SingletonBean's HashCode : {}",bean.hashCode());
            bean.say();
        }
    }    
    

      结果

    2018-12-06 15:10:01,485 INFO :-- [main .. ] o.s.SampleApplication 测试单例bean和原型bean的调用 
    2018-12-06 15:10:01,487 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 376601041 
    2018-12-06 15:10:01,487 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:10:01,487 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 2056499811 
    2018-12-06 15:10:01,487 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:10:01,488 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 890733699 
    2018-12-06 15:10:01,488 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    

      

      通过@Lookup注解实现方法注入

      使用方法注入的方法需要满足以下语法要求

    <public|protected> [abstract] <return-type> theMethodName(no-arguments);
    

      public|protected要求方法必须是可以被子类重写和调用的
      abstract可选,如果是抽象方法,CGLIB的动态代理类就会实现这个方法,如果不是抽象方法,就会覆盖这个方法
      return-type是非单例的类型
      no-arguments不允许有参数

    @Component
    public class SingletonBean {
        private static final Logger logger = LoggerFactory.getLogger(SingletonBean.class);
        
        public void print() {
            PrototypeBean bean = getPrototypeBean().printClass();
            logger.info("Bean SingletonBean's HashCode : {}",bean.hashCode());
            bean.say();
        }
        // 也可以写成 @Lookup("prototypeBean") 来指定需要注入的bean
        @Lookup
        public PrototypeBean getPrototypeBean(){
          reutrn null;
        }
    }

      结果:

      

    2018-12-06 15:18:50,105 INFO :-- [main .. ] o.s.SampleApplication 测试单例bean和原型bean的调用 
    2018-12-06 15:18:50,108 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1349373781 
    2018-12-06 15:18:50,108 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:18:50,108 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1046820071 
    2018-12-06 15:18:50,109 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    2018-12-06 15:18:50,109 INFO :-- [main .. ] o.s.a.SingletonBean Bean SingletonBean's HashCode : 1722645488 
    2018-12-06 15:18:50,110 INFO :-- [main .. ] o.s.a.PrototypeBean say something... 
    

      

  • 相关阅读:
    寻找金秋
    两个周末,两个湖
    桂花林上,再读“六项精进”
    锄奸杜幸,穷寇勿追
    招聘所见思考
    Xufun’s Node.js Primer
    我的软件过程,一年再读
    企业的生命期限,和组织的危机感
    头痛,偷闲,拾黄叶
    喝酒这件事,和等绿灯的习惯
  • 原文地址:https://www.cnblogs.com/wl20200316/p/12850300.html
Copyright © 2011-2022 走看看