AOP是指面向切面编程。编程
在学习AOP以前先来了解一下代理,由于传说中的AOP其实也对代理的一种应用。框架
首先来看这样一段代码:maven
public interface Hello { void say(String name); }
实现类:学习
public class HelloImpl implements Hello { public void say(String name) { System.out.println("Hello!"+name); } }
若是如今须要在say调用的先后作一些操做,那最简单的方法就是直接在say方面中增长代码,但这样确定不够优雅。this
这时候能够用第一种代理模式:静态代理spa
public class HelloProxy implements Hello { private Hello hello; public HelloProxy(){ hello=new HelloImpl(); } public void say(String name) { before(); hello.say(name); after(); } private void before(){ System.out.println("before"); } private void after(){ System.out.println("after"); } }
这样咱们在保留原来HelloImple中say方法的同时,用HelloProxy代理在say方法的先后增长了before和after代理
调用一下试试:code
public static void main(String[] args){ Hello helloProxy=new HelloProxy(); helloProxy.say("Gary"); }
运行结果:对象
那么新的问题来了,若是每次都须要写一个代理类不也很麻烦吗,项目中就会有不少个xxxProxy,所以就出现了 JDK动态代理。blog
代码以下:
public class DynamicProxy implements InvocationHandler { private Object target; public DynamicProxy(Object target){ this.target=target; } public <T> T getProxy(){ return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { before(); Object result=method.invoke(target,args); after(); return result; } private void before(){ System.out.println("before"); } private void after(){ System.out.println("after"); } }
注:Proxy.newProxyInstance()参数有三个:
在该类中target就是被代理的对象,该类实现了InvocationHandler接口,经过反射去动态建立Hello接口的代理类
那么如今的调用就变简单了:
public static void main(String[] args){ DynamicProxy dynamicProxy=new DynamicProxy(new HelloImpl()); Hello helloProxy= dynamicProxy.getProxy(); helloProxy.say("gary"); }
用了JDK动态代理后,咱们以为仍是有问题,若是要代理一个没有接口的类,该怎么办呢,因而就轮到咱们的 CGlib动态代理 出场了。
在Spring、Hibernate等框架中都用到了CGlib动态代理,maven资源以下:
<!-- https://mvnrepository.com/artifact/cglib/cglib --> <dependency> <groupId>cglib</groupId> <artifactId>cglib</artifactId> <version>3.2.6</version> </dependency>
他的使用须要实现MethodInterceptor类,并完成intercept方法。CGlib 提供的是方法级别的代理:
public class CGLibProxy implements MethodInterceptor { public <T> T getProxy(Class<T> cls){ return (T) Enhancer.create(cls,this); } 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("before"); } private void after(){ System.out.println("after"); } }
这样使用时就不须要提供接口了:
public static void main(String[] args){ CGLibProxy cgLibProxy= new CGLibProxy(); Hello hello=cgLibProxy.getProxy(HelloImpl.class); hello.say("Gary"); }
最后再改造一下,加上单例模式就更完美了
private static CGLibProxy instance=new CGLibProxy(); private CGLibProxy(){ }
public static CGLibProxy getInstance(){ return instance; }
调用就改成:
public static void main(String[] args){ Hello hello=CGLibProxy.getInstance().getProxy(HelloImpl.class); hello.say("Gary"); }