Spring Controller全局变量引起的并发问题
2015-11-19 20:33
471 查看
先看下面小段代码,一个controller,一个service。
controller.java代码: ........ @Autowired private XXXService xxxService; ........ @RequestMapping("/doXXX.do") public void doXXX(){ ..... xxxService.saveXXX(String content,....); ..... } XXXService.java代码: private String content; ...... private void init(){//清空请求参数 content = null; ...... } public boolean saveXXX(String content, ......){ this.init(content, ...); this.content = content; //业务逻辑处理 } 以上这段代码在访问量不构成并发时不会出现什么问题。 但当一个请求还未完成,另一个请求已经开始执行的情况下就会出现问题(并发): 第二个请求执行执行init()方法会将第一个请求的content变量设置为null或它本身的值,这样数据就被篡改了。 编码者这样写的目的是因为content等变量需要在多个方法中使用,而且变量很多,但又不想通过方法参数的方式来传递,故使用成员变量。 先看看为什么会出现这种情况。 由于系统采用springmvc框架,springmvc核心控制器DispatcherServlet 默认为每个controller生成单一实例来处理所有用户请求,所以在这个单一实例的controller中,它的XXXService也是一个实例处理所有请求, 这样XXXService的成员变量就被所有请求共享。这样就会出现并发请求时变量内容被篡改的问题。 那么出现这种问题如何解决呢? 第一种方式: 既然是全局变量惹的祸,那就将全局变量都编程局部变量,通过方法参数来传递。 第二种方式: jdk提供了java.lang.ThreadLocal,它为多线程并发提供了新思路。 (当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本) 那么在什么地方使用ThreadLocal呢? 什么变量是请求公用的就将该变量托付给ThreadLocal来管理其线程副本, 所以我们在xxxService中使用它。 XXXService.java代码: private ThreadLocal<String> contentTL = new ThreadLocal<String>(); //private String content;使用contentTL代替content; ...... public boolean saveXXX(String content, ......){ this.contentTL.set(content); //业务逻辑处理 //在各方法中使用content时候用this.contentTL.get()代替 } 此类并发篡改数据的问题,可以在开发工具中设置断点调试的方式来模拟并发。即第一次请求运行到断点时,查看content内容,并且不让程序继续往下运行,同时再发起一个请求,查看content内容。 如内容是第一次请求的内容,并且让第一个请求跑完后,第二个请求到断线处的content正确时,可以确定不会出现并发问题。
相关文章推荐
- Java 去除字符串中的标点
- java.lang.Class.getDeclaredField()方法
- Java关键字及其含义
- 使用myEclipse生成自定义API文档
- Spring的核心技术(二)---容器简介
- java易混淆概念之类变量、成员变量、局部变量
- java中的Vector类
- java 全局按键 swing
- java实现使用url传递paramString和paramStream
- intellij idea 15.01 java: constant string too long
- mybatis-spring最新版下载地址
- Spring 之注解事务 @Transactional
- Spring
- java循环链表实现约瑟夫环问题
- Java实战-山寨QQ
- Java异常处理(二)
- Java学习笔记--堆、栈、常量池
- 创建Java中的Pair类
- 人工鱼群算法java版
- Java中CountDownLatch、CyclicBarrier、Thread.join方法基本应用