springbean解决单例是并发不安全的问题
解决⽅案
1.单例变原型
对web项⽬,可以Controller类上加注解@Scope("prototype")或@Scope("request")
2.线程隔离类ThreadLocal
有⼈想到了线程隔离类ThreadLocal,我们尝试将成员变量包装为ThreadLocal,以试图达到并发安全,代码如下:
@Controller
public class HomeController {
private ThreadLocal<Integer> i = new ThreadLocal<>();
@GetMapping("testsingleton1")
@ResponseBody
public int test1() {
spring framework高危漏洞if (i.get() == null) {
i.set(0);
}
i.().intValue() + 1);
log.info("{} -> {}", Thread.currentThread().getName(), i.get());
().intValue();
}
}
总结:ThreadLocal的⽅式可以达到线程隔离,但还是⽆法达到并发安全。
3.尽量避免使⽤成员变量
有⼈说,单例bean的成员变量这么⿇烦,能不⽤成员变量就尽量避免这么⽤,在业务允许的条件下,将成员变量替换为RequestMapping⽅法中的局部变量
4. 使⽤并发安全的类
Java作为功能性超强的编程语⾔,API丰富,如果⾮要在单例bean中使⽤成员变量,可以考虑使⽤并发安全的容器,如ConcurrentHashMap、ConcurrentHashSet等等等等,将我们的成员变量(⼀般可以是当前运⾏中的任务列表等这类变量)包装到这些并发安全的容器中进⾏管理即可。
spring bean作⽤域有以下5个:
singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作⽤域实例,加上lazy-init就可以避免预处理;
prototype:原型模式,每次通过getBean获取该bean就会新产⽣⼀个实例,创建后spring将不再对其管理;
request:搞web的⼤家都应该明⽩request的域了吧,就是每次请求都新产⽣⼀个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
session:每次会话,同上;
global session:全局的web域,类似于servlet中的application。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论