分享

Java学习路线:Spring框架之动态代理

 好程序员IT 2019-07-18

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

什么是代理模式

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

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


代理模式的优点:

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

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

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

静态代理

什么是静态代理

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

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

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

1. /**

2.  * 卖手机

3.  */

4. public interface SellMobilePhone {

5. 

6.     void sellMobilePhone();

7. }

8. /**

9.  * 小米手机工厂

10.  */

11. public class MiPhoneFactory implements SellMobilePhone{

12. 

13.     public void sellMobilePhone() {

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

15.     }

16. }

17. /**

18.  * 小米代理商

19.  */

20. public class MiPhoneAgent implements SellMobilePhone {

21. 

22.     //被代理者,工厂对象

23.     private SellMobilePhone factory;

24. 

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

26.     public MiPhoneAgent(SellMobilePhone factory){

27.         this.factory = factory;

28.     }

29. 

30.     public void sellMobilePhone() {

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

32.         //调用被代理者的方法

33.         factory.sellMobilePhone();

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

35.     }

36. }

37. public class TestStaticProxy {

38. 

39.     @Test

40.     public void testProxy(){

41.         //创建被代理者

42.         SellMobilePhone factory = new MiPhoneFactory();

43.         factory.sellMobilePhone();

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

45.         //创建代理者

46.         SellMobilePhone agent = new MiPhoneAgent(factory);

47.         //调用卖手机

48.         agent.sellMobilePhone();

49.     }

50. }

静态代理的问题:

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

动态代理

动态代理的特点:

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

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

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

区别是:

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

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

JDK动态代理

实现步骤:

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

2)实现invoke方法

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

51. /**

52.  * 动态代理

53.  */

54. public class SalesAgent implements InvocationHandler{

55. 

56.     //被代理者对象

57.     private Object object;

58. 

59.     /**

60.      * 创建代理对象

61.      * @param object 被代理者

62.      * @return 代理者

63.      */

64.     public Object createProxy(Object object){

65.         this.object = object;

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

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

68.     }

69. 

70.     /**

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

72.      */

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

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

75.         //调用被代理者的方法

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

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

78.         return result;

79.     }

80. }

81. public class TestInvocationHandler {

82. 

83.     @Test

84.     public void testInvocation(){

85.         //创建动态代理对象

86.         SalesAgent agent = new SalesAgent();

87.         //被代理对象

88.         SellMobilePhone sellMobilePhone = new MiPhoneFactory();

89.         //创建代理对象

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

91.         phoneProxy.sellMobilePhone();

92.     }

93. }

CGLib动态代理

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

94. /**

95.  * CGLib动态代理

96.  *

97.  */

98. public class CGLibProxy implements MethodInterceptor {

99. 

100.     /**

101.      * 返回代理对象

102.      * @param object 被代理对象

103.      * @return 代理对象

104.      */

105.     public Object createProxy(Object object){

106.         //创建加强器

107.         Enhancer eh = new Enhancer();

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

109.         eh.setSuperclass(object.getClass());

110.         //设置代理对象的回调

111.         eh.setCallback(this);

112.         return eh.create();

113.     }

114. 

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

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

117.         //调用父类对象的方法

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

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

120.         return res;

121.     }

122. }

总结

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

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多