代理模式(Proxy)的定义:代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象的引用。通俗的来讲代理模式就是我们生活中常见的中介。
为什么要用代理模式?
-
中介隔离作用:在某些情况下,一个客户类不想或者不能直接引用一个委托对象,而代理类对象可以在客户类和委托对象之间起到中介的作用,其特征是代理类和委托类实现相同的接口。
-
开闭原则,增加功能:代理类除了是客户类和委托类的中介之外,我们还可以通过给代理类增加额外的功能来扩展委托类的功能,这样做我们只需要修改代理类而不需要再修改委托类,符合代码设计的开闭原则。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后对返回结果的处理等。代理类本身并不真正实现服务,而是同过调用委托类的相关方法,来提供特定的服务。真正的业务功能还是由委托类来实现,但是可以在业务功能执行的前后加入一些公共的服务。例如我们想给项目加入缓存、日志这些功能,我们就可以使用代理类来完成,而没必要打开已经封装好的委托类。
代理模式最典型的应用就是Spring AOP了。AOP里面的切面就是代理类,切点就是被代理类。
被代理类:
public interface ITarget { void doSomething();}public class Target implements ITarget { public void doSomething() { System.out.println("donSomething..."); }}
下面定义两个代理类: 1.实现权限管理的切面
/** * Created by yzy on 2018/8/22. * * 权限切面 */public class Aspect4Permission implements ITarget{ ITarget target; public Aspect4Permission(ITarget target){ this.target = target; } public void doSomething() { checkPermission(); target.doSomething(); } private void checkPermission(){ System.out.println("checkPermission....."); }}
2.实现日志记录的代理类
/** * Created by yzy on 2018/8/22. * * 日志切面 */public class Aspect4Log implements ITarget{ ITarget target; public Aspect4Log(ITarget target){ this.target = target; } public void doSomething() { printLog(); target.doSomething(); } private void printLog(){ System.out.println("printLog....."); }}
Target target = new Target();Aspect4Log aspect4Log = new Aspect4Log(target);aspect4Log.doSomething();
使用代理模式的前提是被代理类要实现一个接口,这也是spring一直推崇面向接口编程的原因。
为了避免给每一个接口都要生成一个代理类,jdk内置了动态代理来方便开发.
final Target target = new Target();ITarget logAspect = (ITarget) Proxy.newProxyInstance(ITarget.class.getClassLoader(), new Class []{ITarget.class}, new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("printLog.... " ); return method.invoke(target,args); } });logAspect.doSomething();
静态代理总结:
1.可以做到在不修改目标对象的功能前提下,对目标功能扩展. 2.缺点:- 因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多.同时,一旦接口增加方法,目标对象与代理对象都要维护.
如何解决静态代理中的缺点呢?答案是可以使用动态代理方式