今天在学习sprig aop模块的时候,发现aop编程思想在很多地方都有很大的用处。既然aop是一种编程思想,那么我们可不可以不依赖spring ,纯java手写出具有aop思想的代码呢?
在网上找了一篇文章,十分钟理解纯JAVA实现Spring AOP框架,附父亲节短片_搜狐科技_搜狐网,里面用了jdk动态代理的技术,我觉得有必要把看过之后的心得记录一下。
我们先看代码:
上图这是代码结构,除去properties配置文件(为了模仿spring,配置文件而设,其实我们也可以使用其他的方式来得到bean,此处不谈),有两个接口及他们的实现类,一个是advice(切面接口)和advice接口实现类LogAdvice,一个是IBusness(业务接口)及IBusness接口实现类IBusnessImpl,还有BeanFactory(实体类工厂,主要是用于从配置文件中获取信息后生成用于创建代理的类对象)以及ProxyFactoryBean(动态代理类工厂)和一个测试类。
Advice接口及其实现类
Ibusness接口及其实现类
BeanFactory类(重点)
ProxyFactoryBean(重点)
AOPTest类(测试代码)
下面我们来看下BeanFactory类的重点代码:
在调试模式下,我们可以看到代码具体运行过程。通过Class.forName方法获取到class对象,然后通过aClass.newInstance()实例化对象。之后根据配置文件得到IBusnessImpl和LogAdvice的实例化对象。之所以要得到这两个实例对象,因为他们是工厂代理类中的属性。我们看下propertyDeors的值
我们知道了,通过PropertyDescriptor.getWriteMethod()获取实例的set方法,然后调用writeMethod.invoke(bean,advice)将属性值set进去。
接下来我们得到了一个具有完整属性的ProxyFactoryBean实例。以上都是在执行了
ProxyFactoryBean proxyFactoryBean = (ProxyFactoryBean) beanFactory.getBean("bean");这句代码发生的操作
接着在测试类中执行如下代码
然后会发生什么呢?我们接着看代码:
我们在工厂代理类的实例对象中可以发现他的属性就是我们刚刚得到的,通过getProxy方法中这段代码Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);我们得到了业务的实现类(需强转)。
得到业务实现类之后,执行其中的业务方法add,
此时代理类会走invoke方法(有兴趣的可以研究一下InvocationHandler类),这样我们就能在这个方法中实现我们想要插入的操作了(即可以在执行业务方法add之前或之后执行我们想要的操作)。
我们整体回顾一下他的思路:通过BeanFactory中的getbean方法将所需要的代理类的实例初始化好(即填充了他的属性),然后ProxyFactoryBean得到代理类,执行业务方法后调用ProxyFactoryBean中的invoke。整个流程就是这样。
因为这是模仿spring配置文件的做法,我们也可以在ProxyFactoryBean类的getProxy方法中传入类的实例,这样免去了对代理类的属性设置。
代码如下
这样可能更容易理解一些吧。