1.通过PropertyPlaceholderConfigurer在spring中加载其他外部配置文件或者属性文件:
在很多javaEE工程中,Spring的角色非常重要,是一个管理其他模块和组件的轻量级容器,Spring经常需要管理Struts、Ibatis、hibernate等,这些开源框架的配置文件就通过Spring的PropertyPlaceholderConfigurer加载在Spring中进行管理,另外,数据库连接信息、JNDI连接信息属性文件等也可以通过PropertyPlaceholderConfigurer加载到Spring中来管理。用法如下:
(1).通过PropertyPlaceholderConfigurer将其他文件加载到Spring中:
在spring配置文件中添加如下配置:
<bean class=“org.springframework.beans.factory.config.PropertyPlaceholderConfigurer“>
<property name=“locations“>
<value>classpath:要加载的文件名value>
……
property>
bean>
(2).经过(1)中的配置要加载的配置或属性文件就被加载到spring中,如果还需要在运行时使用加载进来的配置或数据文件的一些信息,如使用数据库连接信息或者JNDI连接信息时,就可以使用类型EL表达式的语法进行引用,例如:
<bean id=”dataSource” destroy-method=”close” class=”org.apache.common.dbcp.BasicDataSource”>
<property name=”driverClassName” value=”${driver}”/>
<property name=”url” value=”${url}”/>
<property name=”username” value=”${username}”/>
<property name=”password” value=”${password}”/>
bean>
注意:也可以使用
Proxy.newProxyInstance(类加载器, Class>[]接口数组,回调代理对象(一般是this))
当调用目标对象方法时,通过该方法创建目标对象的代理对象,代理对象会自动调用其invoke方法调用目标对象,并将调用结果返回。
(2).cglib针对普通java类动态代理:
cglib创建动态代理时,不要求目标类必须实现接口,其工作流程如下:
a.动态代理类编写:
Enhancer enhancer = new Enhancer();
//设置目标类的父类为其本身
enhancer.setSuperclass(目标类对象.getClass());
//设置回调对象为动态代理对象本身
enhancer.setCallback(this);
b.实现MethodInterceptor接口:
实现以下方法:
Object intercept(Objectm代理实例,Method代理实例上调用的接口方法的Method 实例,Object[] 传入代理实例上方法调用的参数值的对象数组,MethodProxy 方法代理实例);
注意:cglib不但可以针对类动态代理,还可以针对方法动态代理。
3.面向切面编程(AOP)的基础概念:
以一个普通的java方法来举例
public 返回类型 方法名(参数列表){ ——>环绕通知
方法前处理代码 ——> 前置通知
try{
方法具体实现(方法体)…….
方法后处理代码 ——> 后置通知
}Catch(异常类型 e){
异常处理…… ——> 例外通知
}finally{
最后处理代理…… ——> 最终通知
}
}
a. 横切关注点:如上面5个通知的位置,在java对象中,可以这些具有类似共同处理逻辑的位置加入如权限验证、事物处理、日志记录等处理逻辑的对象称为横切关注点,面向对象编程(OOP)的关注点是纵向将现实世界的事物抽象成编程的对象模型。而面向切面编程(AOP)的关注点是横向的,它将编程对象模型中拥有类似处理逻辑的地方抽象出来形成切面,而编程对象中的处理逻辑就是横切关注点。
b. 切面(Aspect):将横切关注点抽象就形成切面,与类类似,二者关注点不同,类是事物特性的抽象,切面是横切关注点的抽象。
c. 连接点(Joinpoint):被拦截到的点,在Spring中指方法,因为spring只支持方法类型的连接点,即被拦截的方法。如上面例子的方法。
d. 切入点(Pointcut):指对连接点进行拦截的定义,是连接点的集合,即一系列被拦截方法的集合。
e. 通知(Advice):指拦截到连接点之后要做的事情,即拦截之后的逻辑处理。通常的权限验证、事物处理、日志记录等操作就是在通知中定义和完成的。
f. 目标对象(Target):代理的目标对象,即被拦截的对象。如上面例子中方法所在的对象。
g. 织入(Weave):指将切面应用到目标对象,并导致代理对象创建的过程。
h. 引入(Introduction):在不修改代码的前提下,引入可以在运行期为类动态的添加一些方法和字段。
1. Spring中支持面向切面编程(AOP)的依赖包:
Spring解压后目录中的如下3个包:
lib/aspectj/aspectjweaver.jar
lib/aspectj/aspectjrt.jar
lib/cglib/cglib-nodep-2.1-3.jar
2. 在spring中使用面向切面编程(AOP)时,需要在spring配置文件中引入aop的命名空间,即添加如下的配置:
xmlns:aop=”http://www.springframework.org/schema/aop”
“http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.5.xsd”
注意:Spring2.5以后提供两种AOP方法,即基于xml配置文件方式和基于java注解方式。
若要使用注解方式的aop,需要在spring配置文件中添加如下的对象注解方式aop的支持:
<aop:aspectj-autoProxy/>
BeanWrapper 包装类对象 = BeanWrapperImpl(new 被包装类());
包装类对象.setPropertyValue(“属性名”,”属性值”);
通过这种方法就可以给被包装类设置属性。
7. 基于注解方式的面向切面编程(AOP)开发:
(1).在spring配置文件中加入对注解方法的aop支持。
(2).定义切面:
和创建普通类类似,在类前加上”@Aspect”注解,表明该类是一个切面。
(3).在切面中加入切入点:
切入点就是被拦截对象方法的集合,通常切入点定义在切面中某个对切入点进行处理的方法上。使用”@Pointcut”注解,语法如下:
@Pointcut(“execution(* com.test.service..*.*(..))”)
public void anyMethod(){//方法名为切入点名
切入点处理
}
语法参数详解:
a. 第一个”*”:表示被拦截的方法是任意的返回类型。
b. com.test.service:这里是举一个简单的例子,表示要被拦截的包名,即被拦截的包。
c.被拦截包名后面的两个”..”:表示被拦截包下面的子包也递归进行拦截,即被拦截的子包。
d. ”..”之后的”*”:表示被拦截包及其子包下面的所有类,即被拦截的类。
e. 最后一个”*”:表示被拦截类中的所有方法,即被拦截的方法。
f. ”(..)”:表示被拦截的方法接收任意的参数,即被拦截的参数。
注意:切入点定义语法可以支持通配符,但是一定要严格遵循语法规则。如:
@Pointcut(“execution(com.test.service...add*(..))”)
表示对com.test.service包及其子包下所有的类中以”add”开头的方法进行拦截。
(4).在切面中添加通知:
Spring中通知位置请参看3中的小例子。
”@Before”注解:声明前置通知。
“@AfterRutruning”注解:声明后置通知。
“@After”注解:声明最终通知。
“@AfterThrowing”注解:声明例外通知。
“@Around”注解:声明环绕通知。
一个定义通知的例子如下:
@Before(“anyMethod()(切面中声明的切入点名)”)
public void doAccessCheck(){
……
}
注意:环绕通知和其他4种通知的稍有不同,环绕通知的定义方式比较特别,环绕通知在整个方法调用前后都会起作用,因此必须使用连接点对象告诉连接点在环绕通知处理之后继续其逻辑处理。其定义方式如下:
@Around(切入点名)
public Object doBasicProfiling(ProcedingJoinPoint pjp) throws Throwable{
……
return pjp.proceed();//该句是告诉连接点继续执行其他的操作
}
@Before(“切入点名 && args(输入参数名)”)
public void doSomething(String 输入参数名){……}
(2).获取返回结果:
如:
@AfterReturning(Pointcut=”切入点名”,returning=”返回结果名”)
public void dosomething(String 结果名){……}
<aop:config>
<aop:aspect id=”切面id” ref=”spring配置文件中切面类的id”>
<aop:pointcut id=”切入点id”
expression=”execution(* com.test.service..*.*(..))”/>
<aop:before pointcut-ref=”切入点id” method=”切面类中相应的处理方法”/>
<aop:after ……/>
……
aop:aspect>
aop:config>
xmlns:ts=http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-2.5.xsd
b. 在spring配置文件中配置事务管理器如下:
<bean id=”txManager” class=”org.springframework.jdbc.datasource.DataSourceTransactionManager”>
<property name=”dataSource” ref=”spring中配置的数据源bean的id”/>
bean>
c. 在spring配置文件中添加支持注解方式的事务配置项如下:
<tx:annotation-driventransaction-managertx:annotation-driventransaction-manager=”txManager(spring中配置的事务管理器bean的id)”/>
d. 使用基于注解的事务管理:
在Spring所管理的JavaEE工程中,需要使用事务的业务逻辑地方加上“@Transactional”注解。
(2).基于XML文件方式的事务管理:
a. 在spring配置文件中配置事务管理器如下:
<bean id=”txManager” class=”org.springframework.jdbc.datasource.DataSourceTransactionManager”>
<property name=”dataSource” ref=”spring中配置的数据源bean的id”/>
bean>
b. 在spring配置文件中添加事物管理的切面如下:
<aop:config>
<aop:pointcut id=”transactionPointcut”
Expression=”execution(* com.test.service..*.*(..))”/>
<aop:advisor advice-ref=”txAdvice” pointcut-ref=”transactionPointcut”/>
aop:config>
c. 在spring配置文件中为事务通知添加事物处理特性如下:
<tx:advice id=”txAdvice” transactionManager=”txManager”>
<tx:attributes>
<tx:method name=”get*” read-only=”true” propagation=”NOT_SUPPORTED”/>
<tx:method name=”*”/>
tx:attributes>
tx:advice>
<controller>
<set-property property=”processorClass”
Value=”org.springframework.web.struts.DelegatingRequestProcessor”/>
controller>
b. 在Struts配置文件中的Action就不用再写type属性,在Spring配置文件中配置struts的action,配置规则如下:bean的name是struts1配置文件中action的path值。class的值为action的全路径。
注意:在spring中配置的action不再使用id属性,因为id中不允许特殊字符如“/”。
(2).方式2——全权委托方式:
a.在struts1的配置文件中,所有的action的type属性值都写为:org.springframework.web.struts.Delegating.ActionProxy。
b. 在Spring配置文件中配置struts的action,配置规则如下:bean的name是struts1配置文件中action的path值。class的值为action的全路径。
注意:在spring中配置的action不再使用id属性,因为id中不允许特殊字符如“/”。
(3).方式3——struts的action继承spring的ActionSupport方式:
a. struts中的所有action都继承spring的ActionSupport类。
b. 在struts的配置文件中如下配置:
in className=”org.springframework.web.struts.contextLoaderPlugIn”>
<set-property property=”contextConfigLocation” value=”spring配置文件路径”/>
in>
c. 在Struts配置文件中的Action就不用再写type属性,在Spring配置文件中配置struts的action,配置规则如下:bean的name是struts1配置文件中action的path值。class的值为action的全路径。
注意:在spring中配置的action不再使用id属性,因为id中不允许特殊字符如“/”。