深圳Java培训:Spring框架之动态代理

  • Post author:
  • Post category:java


深圳Java培训: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();

}

}

静态代理的问题:

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

动态代理

动态代理的特点:

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

不需要依赖某个具体业务

动态代理分为: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;

}

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

}

总结

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



版权声明:本文为szqf888168原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。