控制反转 IOC:
控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度。其中最常见的方式叫做依赖注入(Dependency Injection,简称DI),还有一种方式叫“依赖查找”(Dependency Lookup)。
在bean.xml中加入bean标签
<!--把对象的创建交给spring来管理--><bean id="accountService" class="cn.figo.service.impl.AccountServiceImpl"></bean><bean id="accountDao" class="cn.figo.dao.impl.AccountDaoImpl"></bean>
然后可以获取spring的Ioc核心容器,并根据id获取bean对象
// 获取核心容器对象ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");// 根据id获取bean对象IAccountService accountService = (IAccountService) ac.getBean("accountService");IAccountDao accountDao = (IAccountDao) ac.getBean("accountDao");
核心容器的继承关系:
ApplicationContext的三个常用实现类:
ClassPathXmlApplicationContext:它可以加载类路径下的配置文件,要求配置文件必须在类路径下。不在的话,加载不了。(更常用)
FileSystemXmlApplicationContext:它可以加载磁盘任意路径下的配置文件(必须有访问权限)
AnnotationConfigApplicationContext:它是用于读取注解创建容器的
核心容器的两个接口
ApplicationContext: 单例对象适用,实际开发通常采用此接口
它在构建核心容器时,创建对象采取的策略是采用立即加载的方式。也就是说,只要一读取完配置文件马上就创建配置文件中配置的对象。
BeanFactory: 多例对象使用
它在构建核心容器时,创建对象采取的策略是采用延迟加载的方式。也就是说,什么时候根据id获取对象了,什么时候才真正的创建对象。
spring对bean的管理
第一种方式:使用默认构造函数创建。
在spring的配置文件中使用bean标签,配以id和class属性之后,且没有其他属性和标签时。
<bean id="accountService" class="cn.figo.service.impl.AccountServiceImpl"></bean>
采用的就是默认构造函数创建bean对象,此时如果类中没有默认构造函数,则对象无法创建。
这里如果我们覆盖掉默认构造函数后,运行会报错
public AccountServiceImpl(String id){System.out.println("create service");}
第二种方式: 使用普通工厂中的方法创建对象(使用某个类中的方法创建对象,并存入spring容器)
通过某一个其他类中的一个方法来创建我们所需要的bean对象,比如这里有一个类InstanceFactory 中的方法 getAccountService()可以创建 IAccountService 对象
/*** 模拟一个工厂类(该类可能是存在于jar包中的,我们无法通过修改源码的方式来提供默认构造函数)*/
public class InstanceFactory {public IAccountService getAccountService(){return new AccountServiceImpl();}
}
在 xml中做如下配置
<bean id="instanceFactory" class="cn.figo.factory.InstanceFactory"></bean>
<bean id="accountService" factory-bean="instanceFactory" factory-method="getAccountService"></bean>
第三种方式:使用工厂中的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
通过某一个其他类中的静态方法来创建我们所需要的bean对象,比如这里有一个类StaticFactory 中的方法 getAccountService()可以创建 IAccountService 对象
/*** 模拟一个工厂类(该类可能是存在于jar包中的,我们无法通过修改源码的方式来提供默认构造函数)*/
public class StaticFactory {public static IAccountService getAccountService(){return new AccountServiceImpl();}
}
在xml中做如下配置:
<bean id="accountService" class="cn.figo.factory.StaticFactory" factory-method="getAccountService"></bean>
这样就可以成功创建相应的bean对象了
bean的作用范围调整
bean标签的scope属性:
作用:用于指定bean的作用范围
取值: 常用的就是单例的和多例的
singleton:单例的(默认值)
prototype:多例的
request:作用于web应用的请求范围
session:作用于web应用的会话范围
global-session:作用于集群环境的会话范围(全局会话范围),当不是集群环境时,它就是session
当设置bean标签的scope为prototype时,是多例的
<bean id="accountService" class="cn.figo.service.impl.AccountServiceImpl" scope="prototype"></bean>
测试一下:
// 获取核心容器对象ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");// 根据id获取bean对象IAccountDao accountDao = (IAccountDao) ac.getBean("accountDao");System.out.println(accountDao);IAccountService accountService = (IAccountService) ac.getBean("accountService");IAccountService accountService2 = (IAccountService) ac.getBean("accountService");System.out.println(accountService);System.out.println(accountService2);System.out.println(accountService == accountService2);
当设置bean标签的scope为singleton时,是单例的
bean对象的生命周期
单例对象:单例对象的生命周期和容器相同
出生:当容器创建时对象出生
活着:只要容器还在,对象一直活着
死亡:容器销毁,对象消亡
多例对象
出生:当我们使用对象时spring框架为我们创建
活着:对象只要是在使用过程中就一直活着。
死亡:当对象长时间不用,且没有别的对象引用时,由Java的垃圾回收器回收
在类中加入 init 和 destroy 方法
public class AccountServiceImpl implements IAccountService {private IAccountDao accountDao = new AccountDaoImpl();public AccountServiceImpl(){System.out.println("对象创建了");}public void saveAccount(){System.out.println("service中的saveAccount方法执行了。。。");}public void init(){System.out.println("对象初始化了。。。");}public void destroy(){System.out.println("对象销毁了。。。");}
}
在 xml中配置 初始化方法和销毁方法
<bean id="accountService" class="cn.figo.service.impl.AccountServiceImpl"scope="singleton" init-method="init" destroy-method="destroy"></bean>
然后测试一下
ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
IAccountService accountService = (IAccountService) ac.getBean("accountService");
accountService.saveAccount();
System.out.println(accountService);
ac.close();
单例时:
多例时: