IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Spring中使用动态代理和AOP来实现事务的管理 -> 正文阅读

[Java知识库]Spring中使用动态代理和AOP来实现事务的管理

一.环境搭建

1)引入相关依赖

 <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.3.14</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
            <version>5.3.14</version>
        </dependency>
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.7</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>5.3.14</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>5.3.14</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.13.2</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.23</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>5.3.14</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>5.3.14</version>
        </dependency>

2)编写jdbc配置

jdbc.driver=com.mysql.cj.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/test?useSSL=false
jdbc.username=root
jdbc.password=root

3)全局配置类

@PropertySource("classpath:jdbc.properties")
@Configuration
@ComponentScan(basePackages = {"transaction.service","transaction.proxy","transaction.config"})
@Import(JdbcConfig.class)
public class GlobalConfig {
}

二.实现思路

1)事务基本实现步骤

  1. 开始事务
  2. 提交事务
  3. 若出现异常回滚事务
  4. 关闭连接

伪代码

try{
beginTransaction();
CRUD
commitTransaction();
}catch(Exception e){
rollbackTransaction();
}finally{
closeTransaction();
}

要实现上诉功能最容易想到的方法就是每一个需要事务的方法里面都写一遍类似上诉代码,显然这是一种糟糕的设计,代码冗余太大。

2)方案实现

  1. Proxy动态代理
  2. AOP
  3. @Transaction注解

后面两种其实都是通过动态代理来实现的

三.案例

数据库表数据如下,要实现的目标就是刘备曹操孙权向自己手下员工发工资转账,这是一个事务的典型应用场景。
在这里插入图片描述
1)编写一个JavaBean类已经相关业务方法

// 省略set、get
public class Account {
    private int aId;
    private String aName;
    private double aBalance;
    private int cId;
}
public interface AccountService {
    /**
     *
     * @param name 用户名
     * @return 账号信息
     */
    Account getAccountByName(String name);

    /**
     *
     * @param account 账号
     * @return 影响行数
     */
    int updateAccount(Account account);

    /**
     *
     * @param sourceName 转账人
     * @param objectName 收账人
     * @param cash 转账金额
     */
    void transfer(String sourceName,String objectName,double cash);
}
@Service("accountService")
public class AccountServiceImpl implements AccountService {

    @Autowired
    JdbcTemplate jdbcTemplate;
    @Override
    public Account getAccountByName(String name) {
        String querySql="select * from t_account where a_name=?";
        BeanPropertyRowMapper<Account> beanPropertyRowMapper = new BeanPropertyRowMapper<>(Account.class);
        return jdbcTemplate.queryForObject(querySql, beanPropertyRowMapper, name);
    }

    @Override
    public int updateAccount(Account account) {
        String updateSql="update t_account set a_balance=? where a_id=?";
        return  jdbcTemplate.update(updateSql, account.getaBalance(), account.getaId());
    }

    @Override
    public void transfer(String sourceName, String objectName, double cash) {
        // 1.得到转账人的信息
        Account A = getAccountByName(sourceName);
        // 2.得到收账人的信息
        Account B = getAccountByName(objectName);
        // 3.修改转账人的金额
        A.setaBalance(A.getaBalance()-cash);
        // 4.修改收账人的金额
        B.setaBalance(B.getaBalance()+cash);
        // 5.更新转账人的信息
        updateAccount(A);
        // 6.更新收账人的信息
        updateAccount(B);
    }
}

2)配置数据源

最主要解决的是数据库连接问题需要保证每一个线程使用的是不同的数据库连接,可以使用ThreadLocal来保证每一个线程拥有不同的连接。不过spring已经帮我们做好了。

 // 1.初始化事务同步管理器
TransactionSynchronizationManager.initSynchronization();
// 2.使用spring的数据源工具类获取当前线程的连接
return DataSourceUtils.getConnection(dataSource);
public class JdbcConfig {
    @Value("${jdbc.driver}")
    private String driver;
    @Value("${jdbc.url}")
    private String url;
    @Value("${jdbc.password}")
    private String password;
    @Value("${jdbc.username}")
    private String username;

    @Bean
    public JdbcTemplate jdbcTemplate(DataSource dataSource){
        return new JdbcTemplate(dataSource);
    }
    @Bean
    public DataSource dataSource(){
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName(driver);
        dataSource.setUrl(url);
        dataSource.setPassword(password);
        dataSource.setUsername(username);
        return dataSource;
    }
    @Bean
    public Connection connection(DataSource dataSource){
        // 1.初始化事务同步管理器
        TransactionSynchronizationManager.initSynchronization();
        // 2.使用spring的数据源工具类获取当前线程的连接
        return DataSourceUtils.getConnection(dataSource);
    }
}

3)封装事务相关方法

@Component
public class TransactionManger {
    @Autowired
    private Connection connection;

    public void beginTransaction(){
        try {
            connection.setAutoCommit(false);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    public void commitTransaction(){
        try {
            connection.commit();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    public void rollbackTransaction(){
        try {
            connection.rollback();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    public void closeTransaction(){
        try {
            connection.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

4)编写一个代理类来实现对AccountService的代理增强

@Component
public class AccountServiceProxy {

    @Qualifier("accountService")
    @Autowired
    private AccountService accountService;

    @Autowired
    private TransactionManger transactionManger;

    @Bean("proxyAccountService")
    public AccountService getAccountServiceProxy(){
        ClassLoader classLoader = accountService.getClass().getClassLoader();
        Class<?>[] interfaces = accountService.getClass().getInterfaces();
        Object proxyInstance = Proxy.newProxyInstance(classLoader, interfaces, (proxy, method, args) ->{
            Object val=null;
            try {
                transactionManger.beginTransaction();
                val = method.invoke(accountService,args);
                transactionManger.commitTransaction();
            }catch (Throwable throwable){
                transactionManger.rollbackTransaction();
                throwable.printStackTrace();
            }finally {
                transactionManger.closeTransaction();
            }
            return val;
        });
        return (AccountService)proxyInstance;
    }
}

5)测试类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = GlobalConfig.class)
public class TestTransaction {

    @Qualifier("proxyAccountService")
    @Autowired
    private AccountService accountService;

    @Test
    public void getAccountByName(){
        Account account = accountService.getAccountByName("刘备");
        System.out.println(account);
    }
    @Test
    public void updateAccount() throws Exception{
        Account account = new Account();
        account.setaId(1);
        account.setaBalance(1100);
        int result = accountService.updateAccount(account);
        System.out.println(result);
    }
    @Test
    public void testTransfer() throws Exception{
        String A="刘备";
        String B="关羽";
        double cash=100d;
        accountService.transfer(A,B,cash);
    }
}

上面是自己通过代理来实现事务操作,可以通过AOP来改造上面代码

    @Autowired
    private TransactionManger transactionManger;

    @Pointcut(value = "execution(* transaction.service.impl.AccountServiceImpl.transfer(..))")
    public void pointcut(){}
    

    @Before(value = "pointcut()")
    public void beforeAdvice(){
        System.out.println("事务开始===>");
        transactionManger.beginTransaction();
    }
    @AfterReturning(value = "pointcut()")
    public void afterReturning(){
        System.out.println("提交事务===>");
        transactionManger.commitTransaction();
    }
    @AfterThrowing(value = "pointcut()")
    public void afterThrowing(){
        System.out.println("回滚事务===>");
        transactionManger.rollbackTransaction();
    }
    @After(value = "pointcut()")
    public void afterAdvice(){
        System.out.println("关闭事务===>");
        transactionManger.closeTransaction();
    }

使用环绕通知:

 @Around(value = "pointcut()")
    public Object transaction(ProceedingJoinPoint pjp){
        try {
            System.out.println("事务开始===>");
            transactionManger.beginTransaction();
            Object obj = pjp.proceed(pjp.getArgs());
            System.out.println("提交事务===>");
            transactionManger.commitTransaction();
            return obj;
        }catch (Throwable throwable){
            throwable.printStackTrace();
            System.out.println("回滚事务===>");
            transactionManger.rollbackTransaction();
        }finally {
            System.out.println("关闭事务===>");
            transactionManger.closeTransaction();
        }
        return null;
    }

当然spring已经将所有的事务操作封装到@Transaction注解中了,不过Spring的注解也是通过AOP的环绕通知来实现的。

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-02-26 11:16:56  更:2022-02-26 11:17:09 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 11:40:27-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码