@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

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





posted @ 2020-07-30 15:20  紫月java  阅读(8063)  评论(0编辑  收藏  举报