Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 500|回复: 0

[Java框架学习]Spring中bean的作用域

[复制链接]
  • TA的每日心情
    开心
    2021-3-12 23:18
  • 签到天数: 2 天

    [LV.1]初来乍到

    发表于 2014-11-6 00:01:02 | 显示全部楼层 |阅读模式
    如何使用spring的作用域:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "singleton"/
    >



    这里的
    scope
    就是用来配置
    spring bean
    的作用域,它标识
    bean
    的作用域。






          在
    spring2.0之前bean只有2种作用域即:singleton(单例)、non-singleton(也称prototype), Spring2.0以后,增加了session、request、global session三种专用于Web应用程序上下文的Bean。因此,默认情况下Spring2.0现在有五种类型的Bean。当然,Spring2.0对Bean的类型的设计进行了重构,并设计出灵活的Bean类型支持,理论上可以有无数多种类型的Bean,用户可以根据自己的需要,增加新的Bean类型,满足实际应用需求。




    1、singleton
    作用域


    当一个
    bean的
    作用域设置为
    singleton, 那么Spring IOC容器中只会存在一个共享的bean实例,并且所有对bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。换言之,当把一个bean定义设置为singleton作用域时,Spring IOC容器只会创建该bean定义的唯一实例。这个单一实例会被存储到单例缓存(singleton cache)中,并且所有针对该bean的后续请求和引用都将返回被缓存的对象实例,这里要注意的是singleton作用域和GOF设计模式中的单例是完全不同的,单例设计模式表示一个ClassLoader中只有一个class存在,而这里的singleton则表示一个容器对应一个bean,也就是说当一个bean被标识为singleton时候,spring的IOC容器中只会存在一个该bean。






    配置实例:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "singleton"/
    >


    或者


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    singleton=
    "true"/
    >






    2、prototype


          prototype作用域部署的bean,每一次请求(将其注入到另一个bean中,或者以程序的方式调用容器的
    1. getBean()
    复制代码
    方法)都会产生一个新的bean实例,相当一个new的操作,对于prototype作用域的bean,有一点非常重要,那就是Spring不能对一个prototype bean的整个生命周期负责,容器在初始化、配置、装饰或者是装配完一个
    prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了。不管何种作用域,容器都会调用所有对象的初始化生命周期回调方法,而对prototype而言,任何配置好的析构生命周期回调方法都将不会被调用。清除prototype作用域的对象并释放任何prototype bean所持有的昂贵资源,都是客户端代码的职责。(让Spring容器释放被singleton作用域bean占用资源的一种可行方式是,通过使用bean的后置处理器,该处理器持有要被清除的bean的引用。)




    配置实例:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "prototype"/
    >


    或者


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    singleton=
    "false"/
    >


      


    3、request


        request表示该针对每一次
    HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP request内有效。


    request、
    session、global session使用的时候首先要在web.xml中做如下配置:



          如果你使用的是
    Servlet 2.4及以上的web容器,那么你仅需要在web应用的XML声明文件web.xml中增加下述ContextListener即可:


      
      
       
       <
       web-app
       >
       
       ...
       
       <
       listener
       >
       

       <
       listener-class
       >
       org.springframework.web.context.request.RequestContextListener
       </
       listener-class
       >
       
       
       </
       listener
       >
       
       ...

       </
       web-app
       >
       

       
      

    ,如果是Servlet2.4以前的web容器,那么你要使用一个javax.servlet.Filter的实现:


      
      
       <
       web-app
       >
       
      ..
      
       <
       filter
       >
       
         
       <
       filter-name
       >
       requestContextFilter
       </
       filter-name
       >
       
         
       <
       filter-class
       >
       org.springframework.web.filter.RequestContextFilter
       </
       filter-class
       >
       
      
       </
       filter
       >
       
      
       <
       filter-mapping
       >
       
         
       <
       filter-name
       >
       requestContextFilter
       </
       filter-name
       >
       
         
       <
       url-pattern
       >
       /*
       </
       url-pattern
       >
       
      
       </
       filter-mapping
       >
       
       ...

       </
       web-app
       >
       

       
      

    接着既可以配置
    bean的作用域了:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "request"/
    >






    4、session


         session作用域表示该针对每一次
    HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP session内有效,配置实例:


    配置实例:


    和request配置实例的前提一样,配置好web启动文件就可以如下配置:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "session"/
    >






    5、global session


         global session作用域类似于标准的
    HTTP Session作用域,不过它仅仅在基于portlet的web应用中才有意义。Portlet规范定义了全局Session的概念,它被所有构成某个portlet web应用的各种不同的portlet所共享。在global session作用域中定义的bean被限定于全局portlet Session的生命周期范围内。如果你在web中使用global session作用域来标识bean,那么web会自动当成session类型来使用。


    配置实例:


    和request配置实例的前提一样,配置好web启动文件就可以如下配置:


    <
    bean
    id
    =
    "role"
    class
    =
    "spring.chapter2.maryGame.Role"
    scope=
    "global session"/
    >






    6、自定义bean装配作用域


           在spring2.0中作用域是可以任意扩展的,你可以自定义作用域,甚至你也可以重新定义已有的作用域(但是你不能覆盖singleton和prototype),spring的作用域由接口org.springframework.beans.factory.config.Scope来定义,自定义自己的作用域只要实现该接口即可,下面给个实例:


    我们建立一个线程的scope,该scope在表示一个线程中有效,代码如下:


      
      
       
       publicclass MyScope
       implements
        Scope
       ...
       {
         privatefinal ThreadLocal threadScope = new ThreadLocal() ...{
              protected Object initialValue() ...{
                 returnnew HashMap();
               }
         };
         public Object get(String name, ObjectFactory objectFactory) ...{
             Map scope = (Map) threadScope.get();
             Object object = scope.get(name);
            if(object==null) ...{
               object = objectFactory.getObject();
               scope.put(name, object);
             }
             return object;
          }
         public Object remove(String name) ...{
             Map scope = (Map) threadScope.get();
             return scope.remove(name);
          }
         publicvoid registerDestructionCallback(String name, Runnable callback) ...{
          }
        public String getConversationId() ...{
            // TODO Auto-generated method stub
            returnnull;
         }
               }
       
      




      
      
       
       

         
       

         
       
      
    复制代码
    回复

    使用道具 举报

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

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-25 11:28 , Processed in 0.400344 second(s), 50 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表