Spring 创建Bean Spring框架花式创建Bean的n种方法(小结)

软件发布|下载排行|最新软件

当前位置:首页IT学院IT技术

Spring 创建Bean Spring框架花式创建Bean的n种方法(小结)

郭乾亮1998.   2021-04-21 我要评论

常用的从容器中获取bean实例使用这样的方式:

@Test
	public void test() {
		Persion p = (Persion) ioc.getBean("p1");
		System.out.println(p);
	}

常用的在容器中配置组件使用这样的方式:

	<bean id="p1" class="com.gql.bean.Persion">
		<property name="name" value="张三"></property>
		<property name="age" value="18"></property>
		<property name="email" value="zs@163.com"></property>
		<property name="gender" value="男"></property>
	</bean>

下面的实验介绍一些Spring容器中注册组件对象的其他方法。

实验1:根据bean的类型从ioc容器中获取实例

@Test
	public void test01() {
		Persion p = ioc.getBean(Persion.class);
		System.out.println(p);
	}

这种方法查找的好处是不需要类型转换,但是如果ioc容器中要找的bean有多个,使用这种方法查找就会报错。可以改用下面的方式:

	@Test
	public void test01() {
		Persion p = ioc.getBean("p1", Persion.class);
		System.out.println(p);
	}

实验2:通过有参构造器为bean的属性赋值

需要提前在bean中添加有参构造器,才能进行下面的测试。

	<bean id="p2" class="com.gql.bean.Persion">
		<constructor-arg name="name" value="李四"></constructor-arg>
		<constructor-arg name="age" value="22"></constructor-arg>
		<constructor-arg name="email" value="ls@163.com"></constructor-arg>
		<constructor-arg name="gender" value="男"></constructor-arg>
	</bean>

使用这种有参构造器为bean的属性赋值,可以省略name,但是value的顺序必须与bean中的顺序一致。(若再使用index和type进行索引,可以不按顺序)

通过名称空间为bean赋值:

添加p命名空间标签头:xmlns:p=“http://www.springframework.org/schema/p”

<bean id="p4" class="com.gql.bean.Persion" p:name="小王" p:age="22"
		p:gender="男" p:email="wang@163.com"></bean>

实验3:为各种属性赋值

引用类型、集合类型、级联类型。

如题,给出一个赋值的Bean对象,为其在容器中注册。此时所有复杂的赋值都在property标签体内。

public class Persion {
	private String name;
	private String gender;
	private Integer age;
	private String email;

	private Car car;

	private List<Book> book;

	private Map<String, Object> maps;

	private Properties properties;
	//省略setter与getter方法
}

普通属性赋值:

<!-- 普通属性赋值 -->
		<property name="name" value="张三"></property>
		<property name="gender" value="男"></property>
		<property name="age" value="20"></property>
		<property name="email">
			<null />
		</property>

引用类型赋值:

	<bean id="p1" class="com.gql.bean.Persion">
		<!-- 引用外部bean -->
		<!-- <property name="car" ref="c1"></property> -->
		<!-- 引用内部bean(内部bean不能被获取到) -->
		<property name="car">
			<bean class="com.gql.bean.Car">
				<property name="carName" value="自行车"></property>
				<property name="color" value="黑色"></property>
				<property name="price" value="400"></property>
			</bean>
		</property>
	</bean>
	
	<bean id="c1" class="com.gql.bean.Car">
		<property name="carName" value="宝马"></property>
		<property name="color" value="白色"></property>
		<property name="price" value="30000"></property>
	</bean>

集合类型赋值:

list

<bean id="p1" class="com.gql.bean.Persion">
		<property name="book">
			<list>
				<bean class="com.gql.bean.Book" p:bookName="西游記" p:author="吴承恩"></bean>
				<ref bean="book01" />
			</list>
		</property>
	</bean>
	
	<bean id="book01" class="com.gql.bean.Book">
		<property name="bookName" value="水浒传"></property>
		<property name="author" value="施耐庵"></property>
	</bean>

map

<property name="maps">
			<map>
				<entry key="k01" value="v01"></entry>
				<entry key="k02" value="v02"></entry>
				<entry key="k03" value-ref="book01"></entry>
				<entry key="k04">
					<bean class="com.gql.bean.Car">
						<property name="carName" value="捷豹"></property>
						<property name="color" value="红色"></property>
						<property name="price" value="50000"></property>
					</bean>
				</entry>
			</map>
		</property>

properties

	<property name="properties">
			<props>
				<prop key="username">root</prop>
				<prop key="password">123456</prop>
			</props>
		</property>

实验4:通过继承实现bean配置信息的重用

下面的代码中p4继承了p3,需要改动的属性在property标签中修改即可,其余的全部原样继承。

<bean id="p3" class="com.gql.bean.Persion">
		<property name="name" value="张三"></property>
		<property name="age" value="20"></property>
		<property name="gender" value="男"></property>
		<property name="email" value="zs@163.com"></property>
	</bean>

	<bean id="p4" parent="p3">
		<property name="name" value="李四"></property>
	</bean>

实验5:单实例singleton和多实例prototype

单实例singleton 多实例prototype
容器启动时创建好对象并保存在容器中 获取Bean时才会创建这个对象
②调用初始化方法 ②调用初始化方法
容器关闭时调用销毁方法 容器销毁时不调用销毁方法
任何时间获取都是获取之前创建好的那个对象 每次获取都会创建一个新的对象

详情可参照博客:通过工厂创建Bean的三种方式

实验6:创建带有生命周期的Bean

ioc容器中注册的组件:

  • 单实例:容器启动的时候就会创建好,容器关闭也会销毁创建的bean。
  • 多实例:获取的时候才创建。

可以为bean自定义一些生命周期方法,spring在创建或销毁的时候就会调用指定的方法。

(1)单实例Singleton测试

在Book类中创建方法:

package com.gql.bean;
public class Book {
	private String bookName;
	private String author;

	public void myInit() {
		System.out.println("Book的初始化方法...");
	}

	public void myDestory() {
		System.out.println("Book的销毁方法...");
	}

	public Book() {
		super();
		// TODO Auto-generated constructor stub
		System.out.println("Book创建...");
	}
}

在xml中注册组件:

<bean id="book01" class="com.gql.bean.Book" destroy-method="myDestory" init-method="myInit"></bean>

测试:

在ApplicationContext中没有close方法,需要将容器类型转换为ConfigurableApplicationContext 。

public class IoCTest {
	ConfigurableApplicationContext ioc = new ClassPathXmlApplicationContext("ApplicationContext.xml");

	@Test
	public void test10() {
		System.out.println("容器关闭了");
		ioc.close();
	}
}


(2)多实例prototype测试

只需要改变xml中注册组件为多实例:

<bean id="book01" class="com.gql.bean.Book" destroy-method="myDestory" init-method="myInit" scope="prototype"></bean>

仍然使用上面的方法进行测试:


可以看到容器的创建销毁一系列都没有进行,这是因为多实例在获取bean的时候才创建实例。

多实例测试中增加获取bean:

	@Test
	public void test10() {
		Object bean = ioc.getBean("book01");
		System.out.println(bean);
		System.out.println("容器关闭了");
		ioc.close();
	}


测试结果中,成功创建了实例,但是容器关闭并没有销毁Bean。

实验7:测试Bean的后置处理器

后置处理器有一点代理对象的意思,使用后置处理器,Bean的生命周期变成下面的样子:

容器启动—>后置处理器Before—>初始化方法—>后置处理器After—>容器关闭(调用销毁方法)

不管有没有初始化方法,后置处理器都会默认其有,继续工作。

后置处理器:

package com.gql.bean;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
/**
 * 1.编写后置处理器 
 * 2.将后置处理器注册在配置文件
 * @author guoqianliang
 *
 */
public class MyBeanPostProcessor implements BeanPostProcessor {

	/**
	 * 初始化前调用
	 */
	@Override
	public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
		// TODO Auto-generated method stub
		System.out.println("Before:" + beanName + "将要调用初始化方法了..." + bean);
		return bean;
	}

	/**
	 * 初始化后调用
	 */
	@Override
	public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
		// TODO Auto-generated method stub
		System.out.println("After:" + beanName + "初始化方法调用完了" + bean);
		return bean;
	}
}

将后置处理器注册在配置文件:

<?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="book01" class="com.gql.bean.Book" destroy-method="myDestory" init-method="myInit" scope="singleton"></bean>
	
	<!-- 后置处理器:可以在bean的初始化前后调用方法 -->
	<bean id="beanPostProcessor" class="com.gql.bean.MyBeanPostProcessor"></bean>
</beans>

测试:

@Test
	public void test11() {
		Object bean = ioc.getBean("book01");
		System.out.println("容器关闭了");
		ioc.close();
	}

实验8:引用外部文件

在Spring中bean默认都是单实例的,而数据库作为单实例是最好不过的,一个项目就是一个连接池,连接池里面管理很多连接,连接是直接从连接池中拿。可以让Spring帮我们创建连接池对象,管理连接池。

注册连接池第一代

在配置中注册连接池:

<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<property name="user" value="root"></property>
		<property name="password" value="Hudie"></property>
		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/test"></property>
		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
	</bean>

测试:

@Test
	public void test12() throws SQLException {
		// 从容器中拿到连接
		// DataSource bean = (DataSource) ioc.getBean("dataSource");
		DataSource bean2 = ioc.getBean(DataSource.class);
		System.out.println(bean2.getConnection());
	}

成功获取到了这个连接:

注册连接池第二代

在config包下创建一个dbconfig.properties用来保存数据库连接信息。

为了防止配置文件中的key与Spring自己的关键字冲突。可以为key加一个前缀,业内通用的做法是使用jabc.xxx

jdbc.username:root
jdbc.password:Hudie
jdbc.jdbcUrl:jdbc:mysql://localhost:3306/test
jdbc.driverClass:com.mysql.jdbc.Driver

注册数据库连接池:

	<bean id="dataSource"
		class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<property name="user" value="${jdbc.username}"></property>
		<property name="password" value="${jdbc.password}"></property>
		<property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
		<property name="driverClass" value="${jdbc.driverClass}"></property>
	</bean>

测试:

@Test
	public void test12() throws SQLException {
		DataSource bean2 = ioc.getBean(DataSource.class);
		System.out.println(bean2.getConnection());
	}

Copyright 2022 版权所有 软件发布 访问手机版

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 联系我们