Spring是一个十分火热开源框架,而AOP(面向切面编程)则是Spring最重要的概念之一,为了更好的理解和学习AOP的思想,使用核心库来实现一次不失为一个好方法。
首先介绍一下AOP的概念,AOP(Aspect Oriented Programming),即面向切面编程,所谓的面向切面编程,就是从一个横切面的角度去设计代码的思想,传统的OOP思想是用封装继承和多态构造一种纵向的层次关系,但不适合定义横向的关系,而AOP思想则对此进行了很好的补充。
例如日志管理代码往往横向的散布在很多对象层次中,但跟它对应的对象的核心功能可以说是毫无关系,还有很多类似的代码,如权限验证,调试输出,事务处理等,也都是如此,这样的话就不利于代码的复用和管理了。
这时候AOP技术就应运而生了,它利用“横切”技术,深入封装对象的内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于后续的可操作性和可维护性。
那么AOP又是如何实现的呢?
答案是动态代理(关于代理会有另外篇章做详细介绍,这里就不赘述了)。实现动态代理有两种方式,一种是JDK动态代理,一种是CGLib动态代理。
那么分别使用两种方式来做一个简单的栗子。
先设计一个场景,假设我们有一个计算接口ICalculator和实现了该接口的计算器类CalculatorImpl。
public interface ICalculator { //加法运算 public int add(int a,int b); //减法 public int subtract(int a,int b); //乘法 public int multiply(int a,int b); //除法 public int devide(int a,int b); }
public class CalculatorImpl implements ICalculator{ @Override public int add(int a, int b) { return a + b; } @Override public int subtract(int a, int b) { return a - b; } @Override public int multiply(int a, int b) { return a * b; } @Override public int devide(int a, int b) { return a / b; } }
如何在不改动原来计算器类内部代码的情况下记录计算器各个方法使用的总次数呢?
有了动态代理后,其实就很简单了,先创建一个类并实现InvocationHandler接口,覆盖invoke方法,
public class TestHandler implements InvocationHandler { private Object targetObject; private int useTimes; //绑定委托对象,并返回代理类 public Object bind(Object targetObject){ this.targetObject = targetObject; return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),targetObject.getClass().getInterfaces(),this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //do something before(); Object result = method.invoke(targetObject,args); after(); return result; } private void before(){ System.out.println("we can do something before calculate."); } private void after(){ useTimes++; System.out.println("已使用:"+useTimes+"次"); } }
别看代码好像有点多,其实主要的方法就是invoke方法,里面的Object result = method.invoke(targetObject,args);相当于继续用原来的参数执行原来方法。这里的before和after为自定义的函数,可以在目标代码执行前后做一些我们想要做的事情,比如这里的使用次数统计。
在bind方法里,传入目标代理对象,并返回一个代理类实例。接下来我们看看如何使用:
public class TestProxy { public static void main(String[] args) { TestHandler proxy = new TestHandler(); ICalculator calculator = (ICalculator)proxy.bind(new CalculatorImpl()); int result = calculator.add(1,2); System.out.println("result is:"+result); result = calculator.subtract(3,2); System.out.println("result is:"+result); result = calculator.multiply(4,6); System.out.println("result is:"+result); result = calculator.devide(6,2); System.out.println("result is:"+result); } }
我们先定义一个TestHandler,然后通过bind方法来获得一个代理实例,之后我们就可以直接使用这个实例了。运行结果如下:
we can do something before calculate.
已使用:1次
result is:3
we can do something before calculate.
已使用:2次
result is:1
we can do something before calculate.
已使用:3次
result is:24
we can do something before calculate.
已使用:4次
result is:3
这样我们就实现了不修改CalculatorImpl内部代码的情况下对代码进行扩展。
接下来用CGLib的方式来实现一次。
先创建一个类来实现MethodInterceptor接口,并覆盖intercept方法。其他代码跟使用JDK代理大同小异,仅仅是获取代理对象的过程有所差异。
public class CGLibProxy implements MethodInterceptor { private int useTimes; private Object target; public Object getInstance(Object target){ this.target=target; Enhancer enhancer =new Enhancer(); enhancer.setSuperclass(this.target.getClass()); enhancer.setCallback(this); return enhancer.create(); } @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { before(); Object result = methodProxy.invokeSuper(o,objects); after(); return result; } private void before(){ System.out.println("we can do something before calculate."); } private void after(){ useTimes++; System.out.println("已使用:"+useTimes+"次"); } }
测试一下:
public class TestCGLibProxy { public static void main(String[] args) { CGLibProxy cgLibProxy = new CGLibProxy(); ICalculator calculator = (ICalculator) cgLibProxy.getInstance(new CalculatorImpl()); int result = calculator.add(1,2); System.out.println("result is:"+result); result = calculator.subtract(3,2); System.out.println("result is:"+result); result = calculator.multiply(4,6); System.out.println("result is:"+result); result = calculator.devide(6,2); System.out.println("result is:"+result); } }
运行结果如下:
we can do something before calculate.
已使用:1次
result is:3
we can do something before calculate.
已使用:2次
result is:1
we can do something before calculate.
已使用:3次
result is:24
we can do something before calculate.
已使用:4次
result is:3
现在我们得到了同样的结果。(需要导入两个包,cglib-2.2.2.jar asm-3.3.jar)
两种方法各有所长,JDK代理需要先设置一个接口,然后才能实现代理,这是它的缺点,也是它的优点,缺点是这样会麻烦一点,而且无法对那些已经封装好的,没有实现接口的类进行代理,而CGLib代理的方式不需要使用接口。但也正是因为如此,JDK代理的方式仅仅拦截类中覆盖接口的方法,而CGLib则会拦截类的所有方法调用。两者各有利弊,所以需要具体情况具体分析。在Spring中也是混杂使用了两种代理模式。
以上就是Java核心库实现简单的AOP的详细内容,更多关于Java 实现aop的资料请关注小牛知识库其它相关文章!
本文向大家介绍Java计算器核心算法代码实现,包括了Java计算器核心算法代码实现的使用技巧和注意事项,需要的朋友参考一下 在进行一个表达式的计算时,先将表达式分割成数字和字符串然后利用出入栈将分割后的表达式进行中缀转后缀,再将后缀表达式进行计算得到结果(思想在上一篇写过)现在贴下Java语言的代码实现。(学习Java时间不长所以可能会有很多不足的地方,我会改进也欢迎大神可以给我一些意见和建议~谢
本文向大家介绍java实现简单的webservice方式,包括了java实现简单的webservice方式的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家分享了java实现webservice方式的具体代码,供大家参考,具体内容如下 经过测试 jdk1.6.10以下会出现bug 建议使用10以上版本 1、定义接口 2、实现接口 3、定义服务 4、测试服务 5、TestSoap 以上就是本文
In Spring Security 3.0, the contents of the spring-security-core jar were stripped down to the bare minimum. It no longer contains any code related to web-application security, LDAP or namespace confi
本文向大家介绍jquery心形点赞关注效果的简单实现,包括了jquery心形点赞关注效果的简单实现的使用技巧和注意事项,需要的朋友参考一下 html代码 css代码 js代码 以上就是小编为大家带来的jquery心形点赞关注效果的简单实现全部内容了,希望大家多多支持呐喊教程~
本文向大家介绍Android实现简单图库辅助器,包括了Android实现简单图库辅助器的使用技巧和注意事项,需要的朋友参考一下 写在前面 实际开发经常会遇到读取相册或者拍照功能,网上也很非常多图库框架,都各有风格,也因此与自己的项目格格不入。再者,框架API太多,需要的配置太多,还要吃力研究。所以,本人摸索大多图库框架,提炼核心,写一个只提供核心的辅助类,剩下的就可以自行玩耍。 实现步骤 第一步,
本文向大家介绍Java 实现 web服务器的简单实例,包括了Java 实现 web服务器的简单实例的使用技巧和注意事项,需要的朋友参考一下 Java 实现 web服务器的简单实例 实例代码: 感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!