我的日常

登录/注册
您现在的位置:论坛 资料库 JAVA开发 > SpringMVC并发访问的线程安全性问题
总共48086条微博

动态微博

查看: 1285|回复: 0

SpringMVC并发访问的线程安全性问题

[复制链接]
admin    

1244

主题

544

听众

1万

金钱

管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    跳转到指定楼层
    楼主
    发表于 2016-06-25 18:06:13 |只看该作者 |倒序浏览
    springmvc的controller是singleton的(非线程安全的),这也许就是他和struts2的区别吧


    和Struts一样,Spring的Controller默认是Singleton的,这意味着每个request过来,系统都会用原有的instance去处理,这样导致了两个结果:一是我们不用每次创建Controller,二是减少了对象创建和垃圾收集的时间;由于只有一个Controller的instance,当多个线程调用它的时候,它里面的instance变量就不是线程安全的了,会发生窜数据的问题。


    当然大多数情况下,我们根本不需要考虑线程安全的问题,比如dao,service等,除非在bean中声明了实例变量。因此,我们在使用spring mvc 的contrller时,应避免在controller中定义实例变量。
    如:
    1. public class Controller extends AbstractCommandController {
    2. ......
    3. protected ModelAndView handle(HttpServletRequest request,HttpServletResponse response,
    4.                         Object command,BindException errors) throws Exception {
    5. company = ................;
    6. }
    7. protected Company company;
    8. }
    复制代码


    在这里有声明一个变量company,这里就存在并发线程安全的问题。
    如果控制器是使用单例形式,且controller中有一个私有的变量a,所有请求到同一个controller时,使用的a变量是共用的,即若是某个请求中修改了这个变量a,则,在别的请求中能够读到这个修改的内容。。


    有几种解决方法:
    1、在Controller中使用ThreadLocal变量
    2、在spring配置文件Controller中声明 scope="prototype",每次都创建新的controller

    所在在使用spring开发web 时要注意,默认Controller、Dao、Service都是单例的。


    SpringMVC多线程环境中如何保证对象的安全性?
    代码:
    1. @RequestMapping("/user")
    2. @Controller
    3. Class UserController
    4. {
    5.         @Resource
    6.         UserService userService;
    7.        
    8.         @RequestMapping("/add")
    9.         public void testA(User user){
    10.                 userService.add(user);
    11.         }

    12.         @RequestMapping("/get")
    13.         public void testA(int id){
    14.                 userService.get(id);
    15.         }
    16. }

    17. @Service("userService")
    18. Class UserService{

    19.         public static Map<Integer,User> usersCache = new HashMap<String,User>();

    20.         public void add(User user){
    21.                 usersCache.put(user.getId(),user);
    22.         }

    23.         public void get(int id){
    24.                 usersCache.get(id);
    25.         }

    26. }
    复制代码

    此段代码,usersCache对象就是线程不安全的。因为它是静态的全局共享对象。如果有多个线程同时调用add方法,可能会发生用户对象被覆盖的情况,也就是id对应对象不一致,这是多线程编程中最常发生的事情。
    所以,可以使用 Collections 工具同步Map。


    static Map<Integer, Users> usersCache = Collections.synchronizedMap(new HashMap<Integer, Users>());


    研究一下,Spring中的源码,它对常用的开源框架做了大量封装,如,hibernate中的sessionFactory,就使用的是 org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean,而在 AnnotationSessionFactoryBean的父类LocalSessionFactoryBean中,定义了大量的ThreadLocal来保证多线程的安全性。
    1. public class LocalSessionFactoryBean extends AbstractSessionFactoryBean implements BeanClassLoaderAware {

    2.         private static final ThreadLocal<DataSource> configTimeDataSourceHolder =
    3.                         new ThreadLocal<DataSource>();

    4.         private static final ThreadLocal<TransactionManager> configTimeTransactionManagerHolder =
    5.                         new ThreadLocal<TransactionManager>();

    6.         private static final ThreadLocal<Object> configTimeRegionFactoryHolder =
    7.                         new ThreadLocal<Object>();

    8.         private static final ThreadLocal<CacheProvider> configTimeCacheProviderHolder =
    9.                         new ThreadLocal<CacheProvider>();

    10.         private static final ThreadLocal<LobHandler> configTimeLobHandlerHolder =
    11.                         new ThreadLocal<LobHandler>();
    复制代码




    科帮网 1、本主题所有言论和图片纯属会员个人意见,与本社区立场无关
    2、本站所有主题由该帖子作者发表,该帖子作者与科帮网享有帖子相关版权
    3、其他单位或个人使用、转载或引用本文时必须同时征得该帖子作者和科帮网的同意
    4、帖子作者须承担一切因本文发表而直接或间接导致的民事或刑事法律责任
    5、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责
    6、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除并致以最深的歉意
    7、科帮网管理员和版主有权不事先通知发贴者而删除本文


    JAVA爱好者①群:JAVA爱好者① JAVA爱好者②群:JAVA爱好者② JAVA爱好者③ : JAVA爱好者③

    快速回复
    您需要登录后才可以回帖 登录 | 立即注册

       

    关闭

    站长推荐上一条 /1 下一条

    发布主题 快速回复 返回列表 联系我们 官方QQ群 科帮网手机客户端
    快速回复 返回顶部 返回列表