Spring中的Aop

Posted 独钓寒江到酒家

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring中的Aop相关的知识,希望对你有一定的参考价值。

一、Aop的概念,以及改造动机

我们来看一个场景,在我们的service层,我们需要实现事务控制,所有的操作必须在同一个事务范围内;比如转账方法,A账户转账给B账户,需要提供事务支持;下面我们看一段代码:

我们可以看到,业务层的事务控制代码,是和我们业务不相关的,可以抽取出来的公共方法,而且又是所有的业务都需要的。
下面我们开始进行改造;

二、抽取公共方法到代理类中,让代理帮我们实现事务

2.1 改造后的原业务方法

   @Override
    public boolean deleteUserAccount(int id) {
        return userAccountDao.deleteUserAccount(id);
    }

    @Override
    public void transferMoney(String sourceName, String targetName, float amount) {
        try
        {
            System.out.println("transferMoney 执行了...");
            UserAccount source = userAccountDao.getUserAccountByName(sourceName);
            UserAccount target = userAccountDao.getUserAccountByName(targetName);
            source.setMoney(source.getMoney() - amount);
            target.setMoney(target.getMoney() + amount);
            userAccountDao.updateUserAccount(source);
            userAccountDao.updateUserAccount(target);
        }catch (Exception e){
            throw new RuntimeException(e);
        }
    }

通过上面代码我们可以看到,抽离了事务控制;

2.2 编写代理类,增强原方法

package org.study.factory;
import org.study.service.IUserAccountService;
import org.study.util.TransactionManager;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class BeanFactory {
    private  IUserAccountService userAccountService;
    private  TransactionManager txManager ;
    public void setUserAccountService(IUserAccountService userAccountService) {
        this.userAccountService = userAccountService;
    }
    public void setTxManager(TransactionManager txManager) {
        this.txManager = txManager;
    }
    /**
     * 对方法进行代理增强,返回
     * @return IUserAccountService
     */
    public IUserAccountService getUserAccountService() {
        IUserAccountService proxyUserAccountService  = (IUserAccountService) Proxy.newProxyInstance(
                userAccountService.getClass().getClassLoader(),
                userAccountService.getClass().getInterfaces(),
                new InvocationHandler() {
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                            Object returnVal = null;
                            try {
                                System.out.println("newProxyInstance 执行了");
                                txManager.beginTransaction();
                                returnVal = method.invoke(userAccountService, args);
                                txManager.commit();

                            } catch (Exception ex) {
                                System.out.println("newProxyInstance 执行了1");
                                ex.printStackTrace();
                                txManager.rollback();
                            } finally {
                                System.out.println("newProxyInstance 执行了2");
                                txManager.release();
                            }
                            return returnVal;
                    }
                }
        );
        return proxyUserAccountService;
    }
}

我们编写了一个代理类,将事务控制放在里面,这样代理类里所有的方法都实现了事务的控制;

三、验证结果

3.1 验证事务是否执行

未执行方法钱的数据

执行测试代码

    @Test
    public void TestTransfer(){
        proxyUserAccountService.transferMoney("test","test1",200);
    }

查看执行结果:

我们的转账代码正确地执行了,没有任何异常;

3.2 在业务方法中抛出异常,验证事务是否控制

执行测试方法,我们发现抛出了异常

再去查看数据库的结果:

证明我们的方法,得到了事务的控制;

四、总结

1、对于实现了接口的类,可以使用JDK的Proxy类创建动态代理;未实现接口的类,使用CGLib动态代理;
2、动态代理是一种Aop思想,可以横向地抽取方法中公共类,常用的应用场景有:事务控制、日志纪录等。

以上是关于Spring中的Aop的主要内容,如果未能解决你的问题,请参考以下文章

2018.12.24 Spring中的aop演示

每日一学之认识Spring中的AOP

spring中的AOP

Spring中的AOP

Spring中的AOP

Spring中的AOP