Spring框架之动态代理

前言

动态代理是一种常用的设计模式,广泛应用于框架中,Spring框架的AOP特性就是应用动态代理实现的,想要理解AOP的实现原理我们就必须先理解动态代理。

什么是代理模式

代理模式是GOF23设计模式之一,代理模式中存在代理者和被代理者,代理者和被代理者都具有相同的功能,并且代理者执行功能时会附加一些额外的操作

如:手机工厂和代理商都具有卖东西的功能,手机代理商除了帮工厂卖手机外,还能在卖手机前打广告推销,卖手机后还可以进行售后服务。

代理模式的优点:

1)符合开闭原则,不用修改被代理者任何的代码,就能扩展新的功能

2)项目的扩展和维护比较方便

代理模式分为:静态代理和动态代理

静态代理

什么是静态代理

1)代理者和被代理者都实现了相同的接口(或继承相同的父类)

2)代理者包含了一个被代理者的对象

3)调用功能时,代理者会调用被代理者的功能,同时附加新的操作

/**

* 卖手机

*/

public interface SellMobilePhone {

void sellMobilePhone();

}

/**

* 小米手机工厂

*/

public class MiPhoneFactory implements SellMobilePhone{

public void sellMobilePhone() {

System.out.println("生产了小米9手机,卖出去!!");

}

}

/**

* 小米代理商

*/

public class MiPhoneAgent implements SellMobilePhone {

//被代理者,工厂对象

private SellMobilePhone factory;

//通过构造方法传入被代理者

public MiPhoneAgent(SellMobilePhone factory){

this.factory = factory;

}

public void sellMobilePhone() {

System.out.println("打广告,做活动~~~~~~~~~~~~~~~~~");

//调用被代理者的方法

factory.sellMobilePhone();

System.out.println("做售后,做推销~~~~~~~~~~~~~~~~~");

}

}

public class TestStaticProxy {

@Test

public void testProxy(){

//创建被代理者

SellMobilePhone factory = new MiPhoneFactory();

factory.sellMobilePhone();

System.out.println("---------------------------------------");

//创建代理者

SellMobilePhone agent = new MiPhoneAgent(factory);

//调用卖手机

agent.sellMobilePhone();

}

}

静态代理的问题:

静态代理只能适合一种业务,如果有新的业务,就必须创建新的接口和新的代理,如添加卖电脑的接口和电脑工厂,就要创建新的电脑代理类。

动态代理

动态代理的特点:

1) 在不修改原有类的基础上,为原来类添加新的功能

2) 不需要依赖某个具体业务

动态代理分为:JDK动态代理和CGLib动态代理

区别是:

JDK动态代理的被代理者必须实现任意接口

CGLib动态代理不用实现接口,是通过继承实现的

JDK动态代理

实现步骤:

1)代理类需要实现InvocationHandler接口

2)实现invoke方法

3)通过Proxy类的newProxyInstance方法来创建代理对象

/**

* 动态代理

*/

public class SalesAgent implements InvocationHandler{

//被代理者对象

private Object object;

/**

* 创建代理对象

* @param object 被代理者

* @return 代理者

*/

public Object createProxy(Object object){

this.object = object;

//Proxy.newProxyInstance创建动态代理的对象,传入被代理对象的类加载器,接口,InvocationHandler对象

return Proxy.newProxyInstance(object.getClass().getClassLoader(),object.getClass().getInterfaces(),this);

}

/**

* 调用被代理者方法,同时添加新功能

*/

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

System.out.println("销售之前,打广告~~~~~~");

//调用被代理者的方法

Object result = method.invoke(object,args);

System.out.println("销售之后,做售后~~~~~~");

return result;

}

}

public class TestInvocationHandler {

@Test

public void testInvocation(){

//创建动态代理对象

SalesAgent agent = new SalesAgent();

//被代理对象

SellMobilePhone sellMobilePhone = new MiPhoneFactory();

//创建代理对象

SellMobilePhone phoneProxy = (SellMobilePhone) agent.createProxy(sellMobilePhone);

phoneProxy.sellMobilePhone();

}

}

CGLib动态代理

特点:通过继承实现,被代理者必须能被继承,通过被代理类创建子类,子类就是父类的代理。

/**

* CGLib动态代理

*

*/

public class CGLibProxy implements MethodInterceptor {

/**

* 返回代理对象

* @param object 被代理对象

* @return 代理对象

*/

public Object createProxy(Object object){

//创建加强器

Enhancer eh = new Enhancer();

//设置被代理对象的类为父类

eh.setSuperclass(object.getClass());

//设置代理对象的回调

eh.setCallback(this);

return eh.create();

}

public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {

System.out.println("售前~~~~~~CGLIB");

//调用父类对象的方法

Object res = proxy.invokeSuper(obj, args);

System.out.println("售后~~~~~~CGLIB");

return res;

}

}

总结

代理模式分为静态代理和动态代理,静态代理只能代理某一种业务,动态代理可以代理各种业务而不用添加新的代理类,动态代理分为JDK动态代理和CGLib动态代理,JDK动态代理类必须实现某个接口,如果没有实现接口则可以使用CGlib实现。

原文地址:https://www.cnblogs.com/qfchen/p/11201701.html