• @Scope注解的作用详解


    TOC

    @Scope注解的作用详解

    在看公司hbase项目config的配置的时候发现有这个注解,顺手百度了一下....然后发现我基础好差(一般来说,数据库连接还是使用连接池的,这种多例连接数据库太耗资源了)


    参考:

    定义:

    @Scope注解是springIoc容器中的一个作用域,在 Spring IoC 容器中具有以下几种作用域:基本作用域singleton(单例)prototype(多例),Web 作用域(reqeust、session、globalsession),自定义作用域

    • singleton单例模式(默认):全局有且仅有一个实例
    • prototype原型模式:每次获取Bean的时候会有一个新的实例
    • request: request表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP request内有效
    • session :session作用域表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP session内有效
    • global session : global session作用域类似于标准的HTTP Session作用域,不过它仅仅在基于portlet的web应用中才有意义

    直接使用字符串容易出问题,spring有默认的参数:

    • ConfigurableBeanFactory.SCOPE_PROTOTYPE,即“prototype”
    • ConfigurableBeanFactory.SCOPE_SINGLETON,即“singleton”
    • WebApplicationContext.SCOPE_REQUEST,即“request”
    • WebApplicationContext.SCOPE_SESSION,即“session”

    使用

    直接在bean对象方法上增加@Scope注解就可以

    /**
    * 定义一个bean对象
    * @return
    */
    @Scope    //@Scope(value = "prototype")
    @Bean(value = "user0", name = "user0", initMethod = "initUser", destroyMethod = "destroyUser")
    public User getUser() {
        System.out.println("创建user实例");
        return new User("张三", 26);
    }

    @Scope注解默认的singleton实例,singleton实例的意思不管你使用多少次在springIOC容器中只会存在一个实例,演示如下只打印了一次创建实例:

    AnnotationConfigApplicationContext applicationContext2 = new AnnotationConfigApplicationContext(MainConfig.class);
    User bean2 = applicationContext2.getBean(User.class);
    System.out.println("实例1 === "+bean2);
    User bean3 = applicationContext2.getBean(User.class);
    System.out.println("实例2 === "+bean3);
    //运行结果
    创建user实例
    实例1 === User [userName=张三, age=26]
    实例2 === User [userName=张三, age=26]

    若是改为@Scope(value="prototype")

    //运行结果
    创建user实例
    实例1 === User [userName=张三, age=26]
    创建user实例
    实例2 === User [userName=张三, age=26]

    使用场景

    几乎90%以上的业务使用singleton单实例就可以,所以spring默认的类型也是singleton,singleton虽然保证了全局是一个实例,对性能有所提高,但是如果实例中有非静态变量时,会导致线程安全问题,共享资源的竞争。

    当设置为prototype时:每次连接请求,都会生成一个bean实例,也会导致一个问题,当请求数越多,性能会降低,因为创建的实例,导致GC频繁,gc时长增加

    多例

    直接在controller层设置多例

    在controller类上设置多例,正常

    单例调用多例

    在controller是默认的单例,service层是多例的时候,多例失效

    虽然Service是多例的,但是Controller是单例的。如果给一个组件加上@Scope("prototype")注解,每次请求它的实例,spring的确会给返回一个新的。问题是这个多例对象Service是被单例对象Controller依赖的。而单例服务Controller初始化的时候,多例对象Service就已经注入了;当你去使用Controller的时候,Service也不会被再次创建了(注入时创建,而注入只有一次)。

    • 方法1: 不使用@Autowired ,每次调用多例的时候,直接调用bean
    • 方法2:spring的解决方法:设置proxyMode,每次请求的时候实例化

      @Scope注解添加了一个proxyMode的属性,有两个值ScopedProxyMode.INTERFACESScopedProxyMode.TARGET_CLASS,前一个表示表示Service是一个接口,后一个表示Service是一个类。

    @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE, proxyMode = ScopedProxyMode.TARGET_CLASS)
    案例
    • 创建bean
        @Bean
        //@Scope标明模式,默认单例模式.  prototype多例模式
        @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE, proxyMode = ScopedProxyMode.TARGET_CLASS)
        public User hbaseConnection() {
            User user = new User((int) (Math.random() * 100));
            System.out.println("调用了hbaseConnection,User:id:" + user.getId());
            return user;
        }
    • controller引入并调用
    @RestController
    @RequestMapping("/demo1")
    public class DemoController {
        @Autowired
        private User hbaseConnection;
        @RequestMapping("/test")
        public void test(){
            System.out.println("test:user:id:"+hbaseConnection.getId());
            System.out.println("test:user:id2:"+hbaseConnection.getId());
        }
    }

    调用接口,结果为

    调用了hbaseConnection,User:id:7
    test:user:id:7
    调用了hbaseConnection,User:id:2
    test:user:id2:2

    每调用一次注入的对象,就会重新创建一个





  • 相关阅读:
    kafka学习总结010 --- 实际项目中遇到的问题1
    kafka学习总结009 --- HW和LEO
    spring学习总结001 --- IOC控制反转、DI依赖注入
    kafka学习总结008 --- 生产者生产数据流程(参照源码)
    kafka学习总结007 --- 生产者Java API实例
    kafka学习总结006 --- 生产者事务
    kafka学习总结005 --- at-exactly-once语义
    kafka学习总结004 --- 生产者ISR
    kafka学习总结003 --- 生产者分区策略
    计算机基础-1(进制转换)
  • 原文地址:https://www.cnblogs.com/ziyue7575/p/c925cfe466df01c1d352f37da8823946.html
Copyright © 2020-2023  润新知