好程序员Java学习路线之Spring框架之动态代理

好程序员Java学习路线之Spring框架之动态代理,前言:动态代理是一种经常使用的设计模式,普遍应用于框架中,Spring框架的AOP特性就是应用动态代理实现的,想要理解AOP的实现原理咱们就必须先理解动态代理。程序员

什么是代理模式设计模式

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

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

图片描述

代理模式的优势:this

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

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

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

静态代理对象

什么是静态代理blog

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实现。

相关文章
相关标签/搜索