Spring使用指南
1. 容器:
BeanFactory是IOC容器的实际代表者,负责容纳和管理Bean;
2. 配置元数据
元数据有多种形式,如XML和annotation,其在Spring中代表同一个对象BeanDefinition;
3. 加载元数据
可通过多种方式加载元数据,如本地文件系统,ClassPath等;
4. Bean的定义
包括2部分:Bean标签自身的DTD或schema定义,Bean对象的定义;
Bean标签自身的DTD或schema定义:
<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-2.5.xsd "
>
Bean对象的定义:
可以用XML或者注解的方式;
5. 多个Bean定义文件
两种方式:
1. 在初始化容器时提供多个Bean文件的位置或者一个通配符表达的位置;
2. 在一个Bean定义文件中通过<import resource="">来导入文件,注意import必须在bean之前;
6. Bean的别名
用<alias name="bean ID" alias="别名">
7. 内部类的Bean表示
其class属性值为:外部类全限定名$内部类名
8. Bean的实例化
1)构造器
为默认方法
<bean id="" class=""/>
2)静态工厂方法
<bean id="Bean实例Id" class="静态工厂类全限定名" factory-method="静态工厂产生实例的方法名"/>
3)实例工厂方法
<bean id="Bean实例Id" factory-bean="工厂实例Id" factory-method="工厂产生Bean实例的方法名"/>
4)容器
通过Spring容器的getBean(String Id)方法来获取实例;
9. 注入依赖
有2种:构造器注入和setter方法注入;
10. 依赖关系处理步骤:
根据定义Bean的配置创建并初始化BeanFactory实例,此时会验证Bean的配置,包括Bean引用的属性是否指向一个有效的Bean;
对于singleton类型和设为提前实例化的Bean,会与BeanFactory同时创建;
11. 依赖配置
1. 简单属性值的配置方式:
1):
<bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName">
<value>com.mysql.jdbc.Driver</value>
</property>
<property name="url">
<value>jdbc:mysql://localhost:3306/mydb</value>
</property>
<property name="username">
<value>root</value>
</property>
<property name="password">
<value>masterkaoli</value>
</property>
</bean>
2):
<bean id="myDataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mydb"/>
<property name="username" value="root"/>
<property name="password" value="masterkaoli"/>
</bean>
3):属性值对的配置方式
<bean id="mappings" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="properties">
<value>
jdbc.driver.className=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/mydb
</value>
</property>
</bean>
12. 延迟初始化Bean
1)在Bean定义中加入lazy-init="true"
2)在Beans定义中加入<beans default-lazy-init="true"></beans>来全局延迟初始化;
13. 自动装配autowire
1)自动装配启用:
在Beans中添加:<beans autowire="自动装配类型"></beans>
2)自动装配类型:
byName:根据名称自动装配,即属性名与Bean的ID相同且有set方法;
byType:根据类型装配,如果有多个匹配类型,则抛异常;
constructor:与byType相同,提供构造器参数;
autodetect:自动决定使用byType还是constructor;
3)自动装配优缺点:
优点:
减少配置数量,与Java代码同步更新;
缺点:
如装配不明确可能出现一些莫名BUG,对象之间的关联关系不再清晰;
4)Bean排除在自动装配之外
Bean添加autowire-cadidate=false
14. 单例对象的多例属性注入
采用Lookup的方法注入:
<bean id="command" class="fiona.apple.AsyncCommand" scope="prototype"/>
<bean id="commandManager" class="fiona.apple.CommandManager">
<lookup-method name="createCommand" bean="command"/>
</bean>
首先,方法必须是抽象的,如果有实现,动态生成的子类仍然会覆盖该方法;
其次,类和方法都不能是final的;
15. Bean的作用域(Scope)
1)配置Bean作用域
<bean id="command" class="fiona.apple.AsyncCommand" scope="作用域"/>
2)作用域类型:
singleton,prototype,request,session,global session;
3)使用request,session,global session作用域
1. 在Bean定义中加入<aop:scoped-proxy/>
2. servlet2.4及以后的版本,web.xml中加入:
<web-app>
<listener>
<listener-class>org.springframework.web.context.request.RequestContextListener</listener-class>
</listener>
</web-app>
servlet2.3及之前的版本,web.xml加入:
<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>
3.<aop:scoped-proxy/>不能在singleton,prototype作用域的Bean定义中添加;
16. 生命周期回调
分为初始化回调和析构回调;
初始化回调:
1)实现InitializingBean接口
2)配置bean的inti-method="初始化回调方法名"并在该bean中实现该方法;
3)缺省初始化方法,配置beans的default-init-mehtod参数,则会调用所有bean的相应初始化方法;
析构回调:
1)实现DisposableBean接口
2)配置bean的destroy-method="初始化回调方法名"并在该bean中实现该方法;
3)缺省初始化方法,配置beans的default-destroy-mehtod参数,则会调用所有bean的相应初始化方法;
Bean初始化方法执行顺序:
@PostConstructor元注解->afterPropertiesSet()->inti-method,相同方法只执行一次
Bean析构方法执行顺序:
@PreDestroy注解的方法->destroy()->destroy-method
注意:生命周期的初始化回调方法是直接与原生Bean打交道,而非代理对象,故此时所有的AOP都不发生作用;
17. 利用MessageSource实现国际化以及消息参数化(适用于ApplicationContext)
1)配置消息Bean
<bean id="messageSource"
class="org.springframework.context.support.ResourceBundleMessageSource">
<property name="basenames">
<list>
<value>format</value> //为资源文件列表
<value>exceptions</value>
<value>windows</value>
</list>
</property>
</bean>
2)使用方法:
调用MessageSource实例的getMessage()方法,可传入参数代替消息中的占位符;
根据local参数决定使用哪个参数文件;
18. 基于注解的配置(Annotation-based)
1)启用:
<Beans>配置:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd">
<context:annotation-config/>
</beans>
2)@AutoWired
@AutoWired可用于set方法,属性值,构造器方法上;当其用于属性值时,并不需要有对应的set方法;
默认采用byType方式,用@Qualifier时,则采用byName方式;
@Resource
与@AutoWired相同,但默认采用byName方式,有name和type属性,指定某个时,切换成对应的方式;
19. 组件扫描
1)启用组件扫描:
同样要在Beans中引入context的DTD文件以及命名空间
添加<context:component-scan base-package="org.example"/>
2)可扫描的组件标签
@Component,@Repository,@Service,@Controller
3)指定组件作用域
使用@Scope
4)指定Bean名
使用@Qualifier
20. Resource接口:
表示各种资源的抽象;
Resource种类:
UrlResource,ClassPathResource,FileSystemResource,ServletContextResource,InputStreamResource,ByteArrayResource;
21: ResourceLoader接口:
根据资源位置的字符串变量表示返回对应的资源对象;
资源位置的前缀,如classpath:,file:,http:隐式的指定了ResourceLoader对象以及返回的Resouce类型;
22. Resource作为属性来配置
如果程序中有Resource对象,则只要在配置中填写对应的资源位置,则spring在装置时会将资源位置自动转换为Resource对象;
如:<bean id="myBean" class="...">
<property name="template" value="some/resource/path/myTemplate.txt"/>
</bean>
24:classpath:与classpath*:前缀的区别:
1)二者都可以使用通配符来表示路径
2)二者均指定返回ClassPathResource对象
3)classpath:只返回符合要求的第一个对象,而classpath*则返回符合要求的所有Resource对象;
4)classpath*采用ClassLoader.getResource(),故依赖于不同应用服务器的ClassLoader实现,可能会有不同;
25. AOP的XML配置方式:
1)实现了AOP方法的bean类;
2)Beans中引入aop的DTD及命名空间;
3)添加如下配置
<bean id="testInterceptor" class="framework.util.SpringInterceptor"></bean>
<aop:config>
<aop:aspect id="aoptest" ref="testInterceptor">
<aop:pointcut id="cut" expression="execution(* framework.dao.*.*(..))" />
<aop:before pointcut-ref="cut" method="beforeMethod" />
<aop:after pointcut-ref="cut" method="afterMethod" />
<aop:after-returning method="returnMethod"
pointcut-ref="cut" />
<aop:after-throwing method="throwMethod"
pointcut-ref="cut" />
<aop:around method="doBasicProfiling" pointcut-ref="cut"/>
</aop:aspect>
</aop:config>
26. 通知类型:
前置通知,后置通知,异常通知,最终通知,环绕通知;
环绕通知与其它通知的区别:可控制何时调用目标方法
可控制返回对象
后置通知与最终通知的区别:后置通知在方法正常返回时执行
最终通知无论方法正常还是异常都会执行
各种通知都可以通过实现Spring自带接口来获得执行时的更多参数(如对象,方法等参数)
27. AOP的两种方式:
通过AspectJ即注释(annotation)、XMLSchema的方式;
AspectJ与XML的区别:
实现方式:AspectJ通过在通知类中加入匹配的注释来实现,需要在SPRING配置中加<aop:aspectj-autoproxy/>
XML则通过在XML中配置来实现,通知类中不需要加注释;
JDK支持:AspectJ需要JDK5+支持,XML不需要;
习惯:用户更习惯于XML;
28. JDK与Cglib代理
1. JDK采用JDK的动态代理,主要采用的是反射的方法在方法的前后添加AOP;
Cglib则采用的继承的方式,通过在子类方法中覆盖父方法来实现在父方法前后添加AOP;
2. Cglib不能实现对final对象及final和private方法的AOP,而JDK可以;
3. Spring中默认对象只要实现接口,就使用JDK;这样会导致用JDK代理产生的对象无法转为对象本身,产生CastException;
全局启用Cglib代理的方法:
<aop:config proxy-target-class=”true” />
如果是对@AspectJ提供支持,则使用<aop:scoped-proxy proxy-target-class=”true” />
29. 单元测试
Mock对象
org.springframework.mock.jndi包提供了JNDI的Mock对象;
org.springframework.mock.web包提供了Servlet API的Mock对象;
org.springframework.mock.web.portlet提供了Portlet API的Mock对象;
30. 事务:
1. 事务属性:
事务隔离,事务传播,事务超时,只读状态
2. 事务管理:
Spring对于不同的事务,实现了不同的事务管理器,如
对于JDBC是DataSourceTransactionManager,Hibernate是HibernateTransactionManager,全局事务则是JTATransactionManager;
3. 涉及事务处理的封装持久化API:
对于不同的事务,Spring用模板类封装了事务处理的过程,如JDBCTemplate,HibernateTemplate,JdoTemplate;
4. 事务的连接的封装类: 通过DataSourceUtils(JDBC),SessionFactoryUtils(Hibernate),PersistenceManagerFactoryUtils(JDO),确保了事务底层的Connection的一致性,进而保证了事务的一致性;
11. Dao层支持:
为各种DAO技术提供了Support类供扩展,简化了DAO层的处理工作;
JDBCDaoSupport: 需DataSource,提供JDBCTemplate;
HibernateDaoSupport:需SessionFactory,提供HibernateTemplate;
JdoDaoSupport: 需PersistenceManagerFactory,提供JdoTemplate;
JpaDaoSupport: 需EntityManagerFactory,提供JpaTemplate;