spring官方手册给出的配置实例。
--------------------------------------------------------------------------------
让我们假定你所有的服务层类定义在以 'x.y.service'
为根的包内。 为了让service包(或子包)下所有名字以 'Service'
结尾的类的对象拥有默认的事务语义,你可以做如下的配置:
<aop:config> <aop:pointcut id="serviceOperation" expression="execution(* x.y.service..*Service.*(..))"/> <aop:advisor pointcut-ref="serviceOperation" advice-ref="txAdvice"/> </aop:config> <!-- these two beans will be transactional... --> <bean id="fooService" class="x.y.service.DefaultFooService"/> <bean id="barService" class="x.y.service.extras.SimpleBarService"/> <!-- ... and these two beans won't --> <bean id="anotherService" class="org.xyz.SomeService"/> <!-- (not in the right package) --> <bean id="barManager" class="x.y.service.SimpleBarManager"/> <!-- (doesn't end in 'Service') --> <tx:advice id="txAdvice"> <tx:attributes> <tx:method name="get*" read-only="true"/> <tx:method name="*"/> </tx:attributes> </tx:advice>
下面的配置示例演示了两个拥有完全不同的事务配置的bean。
<aop:config> <aop:pointcut id="defaultServiceOperation" expression="execution(* x.y.service.*Service.*(..))"/> <aop:pointcut id="noTxServiceOperation" expression="execution(* x.y.service.ddl.DefaultDdlManager.*(..))"/> <aop:advisor pointcut-ref="defaultServiceOperation" advice-ref="defaultTxAdvice"/> <aop:advisor pointcut-ref="noTxServiceOperation" advice-ref="noTxAdvice"/> </aop:config> <!-- this bean will be transactional (see the 'defaultServiceOperation' pointcut) --> <bean id="fooService" class="x.y.service.DefaultFooService"/> <!-- this bean will also be transactional, but with totally different transactional settings --> <bean id="anotherFooService" class="x.y.service.ddl.DefaultDdlManager"/> <tx:advice id="defaultTxAdvice"> <tx:attributes> <tx:method name="get*" read-only="true"/> <tx:method name="*"/> </tx:attributes> </tx:advice> <tx:advice id="noTxAdvice"> <tx:attributes> <tx:method name="*" propagation="NEVER"/> </tx:attributes> </tx:advice>
在service接口所有的方法上执行的一个业务service方法。这里的定义假设所有的接口都被
放置在service包内,它们的实现被放置在service包的子包内。
如果你按照功能对接口进行分组(例如:包com.xyz.someapp.abc.service,com.xyz.def.service),
则这种情况下这个切点表达式应该是:"execution(* com.xyz.someapp..service.*.*(..))"
/**
* A business service is the execution of any method defined on a service
* interface. This definition assumes that interfaces are placed in the
* "service" package, and that implementation types are in sub-packages.
*
* If you group service interfaces by functional area (for example,
* in packages com.xyz.someapp.abc.service and com.xyz.def.service) then
* the pointcut expression "execution(* com.xyz.someapp..service.*.*(..))"
* could be used instead.
*
* Alternatively, you can write the expression using the 'bean'
* PCD, like so "bean(*Service)". (This assumes that you have
* named your Spring service beans in a consistent fashion.)
*/
@Pointcut("execution(* com.xyz.someapp.service.*.*(..))")
public void businessService() {}
在dao接口上定义的所有方法内执行一个数据访问操作。这个定义假设所有的dao接口定义
在dao包内,实现被放置在了子包内。
/**
* A data access operation is the execution of any method defined on a
* dao interface. This definition assumes that interfaces are placed in the
* "dao" package, and that implementation types are in sub-packages.
*/
@Pointcut("execution(* com.xyz.someapp.dao.*.*(..))")
public void dataAccessOperation() {}
任何一个名字以“set”开始的方法的执行:
execution(* set*(..))
AccountService
接口定义的任意方法的执行:
execution(* com.xyz.service.AccountService.*(..))
在service包中定义的任意方法的执行:
execution(* com.xyz.service.*.*(..))
在service包或其子包中定义的任意方法的执行:
execution(* com.xyz.service..*.*(..))
其他的例子:
--------------------------------------------------------------------------------
两个数据源,两个数据库事务拦截器,两个数据库事物切点。
execution组合表达式表述数据库事务切点:
大部分service类的方法使用数据源txManager-datasourceone,对应事物切点txPointcut-datasourceone,事物拦截器txAdvice-datasourceone;
service层PublishService类的几个方法使用数据源txManager-datasourcetwo,对应事物切点txPointcut-datasourcetwo,事物拦截器txAdvice-datasourcetwo;
一个自定义方法拦截器RuntimeLogInterceptor(拦截每个方法,并记录每个方法的执行日志),拦截切点runtimeLogInterceptorPoint;
<!-- 数据源1事务管理器 --> <bean id="txManager-datasourceone" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="DataSource"/> </bean> <!-- 数据源2事务管理器 --> <bean id="txManager-datasourcetwo" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="srcDataSource"/> </bean> <!-- 数据源1事务拦截器 --> <tx:advice id="txAdvice-datasourceone" transaction-manager="txManager-datasourceone" > <tx:attributes> <tx:method name="get*" read-only="true"/> <tx:method name="*"/> </tx:attributes> </tx:advice> <!-- 数据源2事务拦截器 --> <tx:advice id="txAdvice-datasourcetwo" transaction-manager="txManager-datasourcetwo" > <tx:attributes> <tx:method name="get*" read-only="true"/> <tx:method name="*"/> </tx:attributes> </tx:advice> <!-- aop配置 强制使用cglib代理 --> <aop:config proxy-target-class="true"> <!-- datasourceone 数据库事务管理切点 包含的方法:service包(或子包)下所有名字以 'Service' 结尾的类内所有的方法。 不包含的方法:x.y.service包下PublishService类的getResCategory(..)方法, getEditorResList(..)方法,updateResbackrmd(..)方法 --> <aop:pointcut id="txPointcut-datasourceone" expression="execution(* x.y.service..*Service.*(..)) and !execution(* x.y.service.PublishService.getResCategory(..)) and !execution(* x.y.service.PublishService.getEditorResList(..)) and !execution(* x.y.service.PublishService.updateResbackrmd(..))"/> <!-- datasourcetwo 数据库事务管理切点 包含的方法:x.y.service包PublishService类的getResCategory(..)方法, getEditorResList(..)方法,updateResbackrmd(..)方法。 --> <aop:pointcut id="txPointcut-datasourcetwo" expression="execution(* x.y.service.PublishService.getResCategory(..)) or execution(* x.y.service.PublishService.getEditorResList(..)) or execution(* x.y.service.PublishService.updateResbackrmd(..))"/> <!-- 运行日志拦截点 包含的方法:service包(或子包)下所有名字以 'Service' 结尾的类内所有的方法。 不包含的方法:x.y.service包RuntimeLogService类createRuntimeLogBeforeRequest(..)方法, getRuntimeLog(..)方法,setRuntimeLog(..)方法,completeRuntimeLogAfterRequest(..)方法。 --> <aop:pointcut id="runtimeLogInterceptorPoint" expression="execution(* x.y.service..*Service.*(..)) and !execution(* x.y.service.RuntimeLogService.createRuntimeLogBeforeRequest(..)) and !execution(* x.y.service.RuntimeLogService.getRuntimeLog(..)) and !execution(* x.y.service.RuntimeLogService.setRuntimeLog(..)) and !execution(* x.y.service.RuntimeLogService.completeRuntimeLogAfterRequest(..))"/> <!-- 运行日志拦截 --> <aop:advisor advice-ref="RuntimeLogInterceptor" pointcut-ref="runtimeLogInterceptorPoint"/> <!-- datasourceone,datasourcetwo数据库事务拦截 --> <aop:advisor advice-ref="txAdvice-datasourceone" pointcut-ref="txPointcut-datasourceone"/> <aop:advisor advice-ref="txAdvice-datasourcetwo" pointcut-ref="txPointcut-datasourcetwo"/> </aop:config>
总结一下:
--------------------------------------------------------------------------------
1,pointcut既可以定义在一个接口上面(表示实现该接口的类方法将被拦截),同时也可以定义在一个类上面(无接口的情
况,需要强制使用cglib)。在接口上面定义pointcut时无需关心接口实现类的具体位置,只需要定义被拦截的接口及方
法位置。
2,常见的情况:
x.y.service..*Service.*(..)
x.y.service —— 包“x.y.service”
x.y.service.. —— 包“x.y.service”及其子包例如:“x.y.service.abc”,“x.y.service.def”,“x.y.service.ghi”,“x.y.service.jkl”。。。
*Service —— 定义接口(或没有实现接口的类,需要使用cglib代理)表达式;所有以Service结尾的类或接口,注意不是所有以Service结尾的包名。
*(..) —— 定义方法名,方法参数表达式;任意方法的名称,任意方法参数。
com.xyz.service.*.*(..)
com.xyz.service —— 包“com.xyz.service”
*.*(..) —— 任意接口(或没有实现接口的类,需要使用cglib代理),任意方法,任意参数
在service包下定义的任意方法的执行。
com.xyz.service..*.*(..)
com.xyz.service —— 包“com.xyz.service”
com.xyz.service.. ——包“com.xyz.service”及其子包
*.*(..) —— 任意接口(或没有实现接口的类,需要使用cglib代理),任意方法,任意参数
如果总结的有错误,欢迎大家指正。