@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.INTERFACES
和ScopedProxyMode.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
每调用一次注入的对象,就会重新创建一个