本文已参与「新人创作礼」活动,一起开启掘金创作之路。
1.bean管理
1.普通bean
在配置文件中定义bean类型就是返回类型
2.工程bean就是FactoryBean
在配置文件定义bean类型可以和返回类型不一样
- 创建类,让这个类作为工厂bean,实现接口FactoryBean
- 实现接口里面的方法,在实现的方法中定义返回的bean类型
创建一个不同的返回值
public class MyBean implements FactoryBean<Course> {
//定义返回Bean
@Override
public Course getObject() throws Exception {
Course course = new Course();
course.setCname("张思思");
return course;
}
@Override
public Class<?> getObjectType() {
return null;
}
@Override
public boolean isSingleton() {
return false;
}
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="myBean" class="com.Spring.Collection.MyBean">
</bean>
</beans>
public class Course {
private String cname;//课程名称
public void setCname(String cname) {
this.cname = cname;
}
@Override
public String toString() {
return "Course{" +
"cname='" + cname + '\'' +
'}';
}
}
最终是实现Course的类的对象
@Test
public void myBean() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml");
Course mybean = context.getBean("myBean", Course.class);
System.out.println(mybean);
}
2.Bean的作用域
Spring中,那些组成应用程序的主体及由Spring IoC容器所管理的对象,被称之为bean。简单地讲, bean就是由IoC容器初始化、装配及管理的对象
几种作用域中,request、session作用域仅在基于web的应用中使用(不必关心你所采用的是什么web 应用框架),只能用在基于web的Spring ApplicationContext环境。
Singleton 当一个bean的作用域为Singleton,那么Spring IoC容器中只会存在一个共享的bean实例,并且所有对 bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。Singleton是单例类型,就是 在创建起容器时就同时自动创建了一个bean的对象,不管你是否使用,他都存在了,每次获取到的对象 都是同一个对象。注意,Singleton作用域是Spring中的缺省作用域。要在XML中将bean定义成 singleton,可以这样配置: 1 <bean id="ServiceImpl" class="cn.csdn.service.ServiceImpl" scope="singleton">
测试
@Test
public void test03(){
ApplicationContext context = new
ClassPathXmlApplicationContext("applicationContext.xml");
User user = (User) context.getBean("user");
User user2 = (User) context.getBean("user");
System.out.println(user==user2);
}
Prototype 当一个bean的作用域为Prototype,表示一个bean定义对应多个对象实例。Prototype作用域的bean会 导致在每次对该bean请求(将其注入到另一个bean中,或者以程序的方式调用容器的getBean()方法) 时都会创建一个新的bean实例。Prototype是原型类型,它在我们创建容器的时候并没有实例化,而是 当我们获取bean的时候才会去创建一个对象,而且我们每次获取到的对象都不是同一个对象。根据经 验,对有状态的bean应该使用prototype作用域,而对无状态的bean则应该使用singleton作用域。在 XML中将bean定义成prototype,可以这样配置:
<bean id="account" class="com.foo.DefaultAccount" scope="prototype"/>
或者
<bean id="account" class="com.foo.DefaultAccount" singleton="false"/>
Request 当一个bean的作用域为Request,表示在一次HTTP请求中,一个bean定义对应一个实例;即每个HTTP 请求都会有各自的bean实例,它们依据某个bean定义创建而成。该作用域仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:
<bean id="loginAction" class=cn.csdn.LoginAction" scope="request"/>
针对每次HTTP请求,Spring容器会根据loginAction bean的定义创建一个全新的LoginAction bean实 例,且该loginAction bean实例仅在当前HTTP request内有效,因此可以根据需要放心的更改所建实例 的内部状态,而其他请求中根据loginAction bean定义创建的实例,将不会看到这些特定于某个请求的 状态变化。当处理请求结束,request作用域的bean实例将被销毁。 Session 当一个bean的作用域为Session,表示在一个HTTP Session中,一个bean定义对应一个实例。该作用域 仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:
<bean id="userPreferences" class="com.foo.UserPreferences" scope="session"/>
针对某个HTTP Session,Spring容器会根据userPreferences bean定义创建一个全新的 userPreferences bean实例,且该userPreferences bean仅在当前HTTP Session内有效。与request作 用域一样,可以根据需要放心的更改所创建实例的内部状态,而别的HTTP Session中根据 userPreferences创建的实例,将不会看到这些特定于某个HTTP Session的状态变化。当HTTP Session 最终被废弃的时候,在该HTTP Session作用域内的bean也会被废弃掉。
1.设置创建Bean实例是单实例还是多实例
在Spring下默认是单实例对象
1.在Spring配置文件bean标签里面有属性用于设置单实例还是多实例 scope="" 就是设置多实例还是单实例
- prototype是多实例
- singleton是单实例也是默认实例
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--配置多实例对象-->
<bean id="book" class="com.Spring.IOC.Book" scope="prototype">
<property name="bname" value="java开发"></property>
</bean>
</beans>
public class Book {
private String bname;
public void setBname(String bname) {
this.bname = bname;
}
}
prototype与singleton的区别
- 设置scope是singleton的时候,加载Spring配置文件就会创建单实例对象
- scope是prototype的时候,不是在加载Spring配置文件时候创建对象,在调用getBean方法时候创建多实例对象
6.Bean的生命周期
1.过程
- 通过构造器创建Bean实例(无参构造)
- 为Bean的属性设置值和对其他Bean引用(调用set方法)
- 调用Bean的初始化方法(需要进行配置初始化方法)
- Bean可以使用了(对象获取到了)
- 当容器关闭时,调用Bean的销毁的方法(需要进行配置销毁方法)
public class Orders {
private String oname;
public Orders() {
System.out.println("1.执行无参数构造创建Bean实例");
}
public void setOname(String oname) {
this.oname = oname;
System.out.println("2.调用set方法设置属性值");
}
public void initMethod(){
System.out.println("3.执行初始化的方法");
}
public void destroyMethod(){
System.out.println("5.执行销毁方法");
}
}
测试类
@Test
public void testBean() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml");
Orders orders = context.getBean("orders2",Orders.class);
System.out.println("4.获取创建Bean实例对象");
System.out.println(orders);
//手动销毁Bean实例
((ClassPathXmlApplicationContext) context).close();
}
xml配置里面要加上init-method 和destory-method这两个方法都在Orders类里面创建
<bean id="orders2" class="com.Spring.Collection.Orders" init-method="initMethod" destroy-method="destroyMethod">
<property name="oname" value="手机"></property>
</bean>
2.添加后置处理器后有七步
- 通过构造器创建Bean实例(无参构造)
- 为Bean的属性设置值和对其他Bean引用(调用set方法)
- 把Bean实例传给Bean后置处理器的方法
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
return bean;
}
- 调用Bean的初始化方法(需要进行配置初始化方法)
- 把Bean实例传递给Bean后置处理器的方法
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
return bean;
}
- Bean可以使用了(对象获取到了)
- 当容器关闭时,调用Bean的销毁的方法(需要进行配置销毁方法)
添加后置处理器方法实现BeanPostProcessor,创建后置处理器
public class MyBeanPost implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("在初始化之前执行的方法");
return bean;
}
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("在初始化方法之后执行的方法");
return bean;
}
}
public class Orders {
private String oname;
public Orders() {
System.out.println("1.执行无参数构造创建Bean实例");
}
public void setOname(String oname) {
this.oname = oname;
System.out.println("2.调用set方法设置属性值");
}
public void initMethod(){
System.out.println("3.执行初始化的方法");
}
public void destroyMethod(){
System.out.println("5.执行销毁方法");
}
}
只需要将配置的后置对象加入到配置的文件中,在里面的其他配置实例都会有后置对象
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--配置多实例对象-->
<bean id="orders2" class="com.Spring.Collection.Orders" init-method="initMethod" destroy-method="destroyMethod">
<property name="oname" value="手机"></property>
</bean>
<!--配置后置处理器-->
<bean id="myBeanPost" class="com.Spring.Collection.MyBeanPost">
<--7-->
</bean>
</beans>