您现在的位置是:首页 >学无止境 >Spring03--Spring基于注解的AOP、Spring事务网站首页学无止境
Spring03--Spring基于注解的AOP、Spring事务
简介Spring03--Spring基于注解的AOP、Spring事务
Spring基于注解的AOP
一、基于注解的AOP介绍
- AOP注解方式 和 XML方式完成的功能是一样,只是采用方式不同而已。
- 将原有的XML方式使用注解注意替代
二、环境搭建
- 使用Spring02中Spring基于配置文件的AOP的环境搭建
三、使用注解替换XML的配置步骤
1、通知类使用注解和使用@Aspect注解声明为切面
2、在增强的方法上使用注解配置通知
(1)@Before
- 作用:把当前方法看成是前置通知
- 属性
- value:用于指定切入点表达式,还可以指定切入点表达式的引用。
@Before("execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))")
public void startTransaction() {
try {
System.out.println("启动事务");
connection.setAutoCommit(false);
} catch (SQLException e) {
e.printStackTrace();
}
}
(2) @AfterReturning
- 作用:把当前方法看成是后置通知。
- 属性
- value:用于指定切入点表达式,还可以指定切入点表达式的引用
@AfterReturning("execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))")
public void commitTransaction() {
try {
System.out.println("提交");
connection.commit();
} catch (SQLException e) {
e.printStackTrace();
}
}
(3)@AfterThrowing
- 作用:把当前方法看成是异常通知。
- 属性
- value:用于指定切入点表达式,还可以指定切入点表达式的引用
@AfterThrowing("execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))")
public void rollbackTransaction() {
try {
System.out.println("回滚");
connection.rollback();
} catch (SQLException e) {
e.printStackTrace();
}
}
(4)@After
- 作用:把当前方法看成是最终通知。
- 属性
- value:用于指定切入点表达式,还可以指定切入点表达式的引用
@After("execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))")
public void closeConnection() {
try {
System.out.println("释放资源");
connection.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
3、在Spring 配置文件中开启 Spring 对注解AOP的支持
<!-- 开启 spring 对注解 AOP 的支持 -->
<aop:aspectj-autoproxy/>
- 或者使用 在Spring 配置类中 使用注解@EnableAspectJAutoProxy开启对 注解AOP 的支持
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-h8vxQISn-1683722293179)(F:Java语言课程笔记第六阶段myselfSpringimg使用注解开启对注解AOP的支持.png)]
4、环绕通知直接配置
- @Around
- 作用:把当前方法看成是环绕通知。
- 属性
- value:用于指定切入点表达式,还可以指定切入点表达式的引用。
@Around("execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))")
public Object transactionAround(ProceedingJoinPoint pjp) {
Object result = null;
try {
Object[] args = pjp.getArgs();
startTransaction();
result = pjp.proceed(args);
commitTransaction();
} catch (Throwable throwable) {
rollbackTransaction();
throwable.printStackTrace();
} finally {
closeConnection();
}
return result;
}
5、切入点表达式注解
Spring事务讲解
1、Spring中事务的API详解
(1)PlatformTransactionManager作用
- PlatformTransactionManager是一个 接口
- 它是 Spring 的事务管理器,它里面提供了我们常用的操作事务的方法。
**注意:**PlatformTransactionManager 是接口类型,不同的 Dao 层技术则有不同的实现类,
-
Dao 层技术是jdbc 或 mybatis 时:org.springframework.jdbc.datasource.DataSourceTransactionManager
-
Dao 层技术是hibernate时:org.springframework.orm.hibernate5.HibernateTransactionManager
(2)TransactionDefinition作用
TransactionDefinition 是事务的定义信息对象,里面有如下方法
事务隔离级别
设置隔离级别,可以解决事务并发产生的问题,如脏读、不可重复读和虚读。
事务隔离级别 | 说明 |
---|---|
ISOLATION_DEFAULT | 默认级别,归属下列某一种 |
ISOLATION_READ_UNCOMMITTED | 未提交读,可以读取未提交数据 |
ISOLATION_READ_COMMITTED | 已提交读,只能读取已提交数据,解决脏读问题(Oracle默认级别) |
ISOLATION_REPEATABLE_READ | 可重复读,解决不可重复度问题(MySQL默认级别) |
ISOLATION_SERIALIZABLE | 串行化,解决幻读(虚读)问题 |
事务传播行为
事务传播行为 | 说明 |
---|---|
REQUIRED | 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。一般的选择(默认值) |
SUPPORTS | 支持当前事务,如果当前没有事务,就以非事务方式执行(没有事务) |
MANDATORY | 使用当前的事务,如果当前没有事务,就抛出异常 |
REQUERS_NEW | 新建事务,如果当前在事务中,把当前事务挂起。 |
NOT_SUPPORTED | 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起 |
NEVER | 以非事务方式运行,如果当前存在事务,抛出异常 |
NESTED | 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行 REQUIRED 类似的操作 |
事务超时时间
默认值是-1,没有超时限制。如果有,以秒为单位进行设置。
是否是只读事务
建议查询时设置为只读。
(3)TransactionStatus作用
TransactionStatus 接口提供的是事务具体的运行状态,方法介绍如下。
真正干活: 事务管理器 会根据TransactionDefinition 定义的信息, 进行事务的管理。管理过程当中产生事务的状态保存在TransactionStatus当中。
2、环境搭建
沿用上一小章节的代码,修改AccountDaoImpl 具体使用使用Spring提供的JdbcTemplate模板对象实现
package com.etime.dao.impl;
import com.etime.dao.AccountDao;
import com.etime.entity.Account;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;
@Repository("accountDao")
public class AccountDaoImpl implements AccountDao {
@Autowired
private JdbcTemplate jdbcTemplate;
public Account getAccountByName(String name){
Account account = null;
String sql = "select * from account where name = ?";
account = jdbcTemplate.queryForObject( sql, new BeanPropertyRowMapper<>(Account.class),name);
return account;
}
public int update(Account account){
int rows = 0;
String sql = "update account set money = ? where id = ?";
rows = jdbcTemplate.update(sql, account.getMoney(), account.getId());
return rows;
}
}
3、事务管理配置步骤
(1)配置事务管理器
<bean id="dtm" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="ds"></property>
</bean>
(2) 配置事务的通知引用事务管理器
<!--事务的配置-->
<tx:advice id="d" transaction-manager="dtm">
</tx:advice>
(3)在tx:advice中配置事务的属性
<!--
指定方法名称:是业务核心方法
read-only:是否是只读事务。默认 false,不只读。
isolation:指定事务的隔离级别。默认值是使用数据库的默认隔离级别。
propagation:指定事务的传播行为。
timeout:指定超时时间。默认值为:-1。永不超时。
rollback-for:用于指定一个异常,当执行产生该异常时,事务回滚。产生其他异常,事务不回滚。
没有默认值,任何异常都回滚。
no-rollback-for:用于指定一个异常,当产生该异常时,事务不回滚,产生其他异常时,事务回滚。没有默认值,任何异常都回滚。
-->
<tx:attributes>
<tx:method name="*"/>
</tx:attributes>
(4)配置 AOP 切入点表达式
<!--事务的aop增强-->
<aop:config>
<aop:pointcut id="point"
expression="execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))"/>
</aop:config>
<!--在aop:config标签内部:建立事务的通知和切入点表达式的关系-->
<aop:advisor advice-ref="d" pointcut-ref="point"></aop:advisor>
完整的配置文件内容:
<?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:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<context:component-scan base-package="com.etime"></context:component-scan>
<context:property-placeholder location="classpath:jdbc.properties"></context:property-placeholder>
<bean id="ds" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driver}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="user" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<bean id="jt" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="ds"></property>
</bean>
<bean id="dtm" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="ds"></property>
</bean>
<tx:advice id="d" transaction-manager="dtm">
<tx:attributes>
<tx:method name="*"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="point"
expression="execution(public void com.etime.service.impl.AccountServiceImpl.transferAccount(..))"/>
<aop:advisor advice-ref="d" pointcut-ref="point"></aop:advisor>
</aop:config>
</beans>
(5)测试
package com.etime.test;
import com.etime.service.AccountService;
import com.etime.util.SpringConfig;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import java.sql.SQLException;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:application.xml"})
public class SpringTest {
@Autowired
private ApplicationContext context;
@Test
public void t01() throws SQLException {
AccountService accountService = (AccountService) context.getBean("accountService");
accountService.transferAccount("liudh", "fanbb" , 1000);
}
}
iguration(locations = {“classpath:application.xml”})
public class SpringTest {
@Autowired
private ApplicationContext context;
@Test
public void t01() throws SQLException {
AccountService accountService = (AccountService) context.getBean("accountService");
accountService.transferAccount("liudh", "fanbb" , 1000);
}
}
风语者!平时喜欢研究各种技术,目前在从事后端开发工作,热爱生活、热爱工作。