關於spring 事務 和 AOP 管理事務和打印日志問題
1. <tx:annotation-driven /> 就是支持事務注解的(@Transactional) 。
可以在server層總使用@Transactional,進行方法內的事務管理
配置
<!-- 事務注解-->
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true" />
<!-- 事務管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 數據源 -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName">
<value>oracle.jdbc.driver.OracleDriver</value>
</property>
<property name="url">
<value>jdbc:oracle:thin:@localhost:1521:orcl</value>
</property>
<property name="username">
<value>xxx</value>
</property>
<property name="password">
<value>xxx</value>
</property>
<property name="maxActive">
<value>255</value>
</property>
<property name="maxIdle">
<value>2</value>
</property>
<property name="maxWait">
<value>120000</value>
</property>
</bean>
2. 在applicationContent.xml 引入 applicationContext-aop.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd ">
<!-- aop 統一處理類 -->
<bean id="serviceLogging" class="com.xxx.hrzcyp.aop.ServiceLogging"/>
<!-- aop 配置 -->
<aop:config>
<!-- 切入點 -->
<aop:pointcut id="serviceMethodLoggingPointcut" expression="execution(* com.xxx.service.impl.*.*(..))"/>
<!-- 定義 切面 order 的值越小,說明越先被執行 -->
<aop:aspect id="serviceMethodLoggingAspect" ref="serviceLogging" order="100">
<aop:before method="loggingMethodInvoked" pointcut-ref="serviceMethodLoggingPointcut"/>
<aop:after-throwing method="loggingServiceException" pointcut-ref="serviceMethodLoggingPointcut" throwing="throwable"/>
</aop:aspect>
</aop:config>
</beans>
3. 用於aop處理日志和異常的java
package com.xxx.aop;
import org.aspectj.lang.JoinPoint;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* AOP 統一處理異常和返回值
*/
public class ServiceLogging {
private static Logger exceptionLogger = LoggerFactory.getLogger("ExceptionLogging");
private static Logger methodInvokeLogger = LoggerFactory.getLogger("MethodInvokeLogging");
/**
* 捕獲Service層拋出的異常並做日志
*
* @param throwable 方法拋出的異常
*/
public void loggingServiceException(Throwable throwable) {
if (exceptionLogger.isErrorEnabled()) {
StringBuilder builder = new StringBuilder();
builder.append("cause:").append(throwable.getMessage());
builder.append("\n\tstackTrack:\n");
for (StackTraceElement stack : throwable.getStackTrace()) {
builder.append("\t\t");
builder.append(stack.toString());
builder.append("\n");
}
exceptionLogger.error(builder.toString());
}
}
/**
* 記錄Service方法的調用
*
* @param joinPoint 切入點
*/
public void loggingMethodInvoked(JoinPoint joinPoint) {
if (methodInvokeLogger.isDebugEnabled()) {
String methodName = joinPoint.getSignature().getName();
Object[] args = joinPoint.getArgs();
StringBuilder builder = new StringBuilder();
builder.append("調用方法[AOP切入點]:").append(methodName);
builder.append("\t參數:");
for (Object arg : args) {
builder.append(arg.toString());
}
methodInvokeLogger.debug(builder.toString());
}
}
}
總結
以上工作做完后,只要有訪問當 com.xxx.service.impl 目錄下的所有方法。aop都會幫助我們答應訪問的日志,如果加上 @Transactional 會將整個方法內的事務統一管理。有異常時也會通過aop拋出異常。