1.spring Bean生命周期
Bean的作用域可以通过Bean标签的scope属性进行设置,Bean的作用域包括:
默认情况下scope="singleton",那么该Bean是单例,任何人获取该Bean实例的都为同一个实例;
scope="prototype",任何一个实例都是新的实例;
scope="request",在WEB应用程序中,每一个实例的作用域都为request范围;
scope="session",在WEB应用程序中,每一个实例的作用域都为session范围;
一个典型的Bean配置是这样的:<bean id="user" class="..." scope="singleton"/>,其中的scope表明这个bean是不是单例,默认为单例。
2.Bean可以通过指定属性init-method指定初始化后执行的方法,以及通过指定属性destroy-method销毁时执行的方法。
语法:<bean .... destroy-method="销毁时调用的方法名" init-method="初始化后执行的方法名"/>
3.恶汉模式和懒汉模式
同jvm的单例模式一样,spring容器的单例模式同样分为这两种。默认情况下spring singleton为恶汉模式。
恶汉模式:启动Spring容器时(即实例化容器时),为所有的Spring配置文件中定义的bean都生成一个实例;
懒汉模式:在第一个请求是才生成一个实例,以后的请求都是调用这个实例,可以通过显示的声明将其设置为懒汉模式,如下:
<beans default-lazy-init="true">
4.Spring单例Bean和单例模式的区别
Spring单例Bean和单例模式的最大区别在于他们关联的环境不同,单例模式是指在一个JVM进程中仅有一个实例,二Spring单例是指一个Spring Bean容器(ApplicationContext)中仅有一个实例。
首先看单例模式,在一个JVM进程中(理论上,一个运行的JAVA程序就必定有自己一个独立的JVM)仅有一个实例,所以无论在程序中何处获取单例模式的实例对象,始终都会返回同一个对象,以Java内置的Runtime为例(现在枚举是到单例模式的最佳实现),无论何时何地获取,西面的判断始终为真:
// 基于懒汉模式实现
// 在一个JVM实例中始终只有一个实例
Runtime.getRuntime() == Runtime.getRuntime()
与此相比,Spring的单例Bean是预期容器(ApplicationContext)模切相关的,所以在一个JVM进程中,所以在一个JVM进程中,如果有多个Spring容器,及时是到哪里bean,一定会创建多个实例,代码示例如下:
// 第一个Spring Bean容器
ApplicationContext context_1 = new FileSystemXmlApplicationContext("classpath:/ApplicationContext.xml");
Person yiifaa_1 = context_1.getBean("yiifaa", Person.class);
// 第二个Spring Bean容器
ApplicationContext context_2 = new FileSystemXmlApplicationContext("classpath:/ApplicationContext.xml");
Person yiifaa_2 = context_2.getBean("yiifaa", Person.class);
// 这里绝对不会相等,因为创建了多个实例
System.out.println(yiifaa_1 == yiifaa_2);
下面使其对应的配置文件:
<!-- 即使声明了为单例,只要有多个容器,也一定会创建多个实例 -->
<bean id="yiifaa" class="com.stixu.anno.Person" scope="singleton">
<constructor-arg name="username">
<value>yiifaa</value>
</constructor-arg>
</bean>
所以,总而言之,Spring的单例bean和Spring bean管理容器密切相关,每一个容器都会创建自己独有的实例,所以GOF设计模式中的单例模式相差极大,但是在实际应用中,如果将对象的生命周期完全交给Spring管理(不在其他地方通过new或者反射等方式创建),其实也能打包单例模式的效果。