前言:Proxy是比较有用途的一种模式,而且变种较多,应用场合覆盖从小结构到整个系统的大结构,Proxy是代理的意思,我们也许有代理服务器等概念,代理概念可以解释为:在出发点到目的地之间有一道中间层,意为代理。
1.代理模式的定义
代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象。这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。这种类型的设计模式属于结构型模式。
进群:697699179可以获取Java各类入门学习资料!
这是我的微信公众号【编程study】各位大佬有空可以关注下,每天更新Java学习方法,感谢!
学习中遇到问题有不明白的地方,推荐加小编Java学习群:697699179内有视频教程 ,直播课程 ,等学习资料,期待你的加入
2.作用
中介隔离作用:
在某些情况下,一个客户类不想或者不能直接引用一个委托对象,而代理类对象可以在客户类和委托对象之间起到中介的作用,其特征是代理类和委托类实现相同的接口。
开闭原则,增加功能:
代理类除了是客户类和委托类的中介之外,我们还可以通过给代理类增加额外的功能来扩展委托类的功能,这样做我们只需要修改代理类而不需要再修改委托类,符合代码设计的开闭原则。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后对返回结果的处理等。代理类本身并不真正实现服务,而是同过调用委托类的相关方法,来提供特定的服务。真正的业务功能还是由委托类来实现,但是可以在业务功能执行的前后加入一些公共的服务。例如我们想给项目加入缓存、日志这些功能,我们就可以使用代理类来完成,而没必要打开已经封装好的委托类。
3.优缺点
优点:
1、职责清晰。
2、高扩展性。
3、智能化。
缺点:
1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
4.代理模式的分类
如果按照代理创建的时期来进行分类的话, 可以分为两种:静态代理、动态代理。静态代理是由程序员创建或特定工具自动生成源代码,在对其编译。在程序员运行之前,代理类.class文件就已经被创建了;动态代理是在程序运行时通过反射机制动态创建的。
5.代码实现
5.1静态代理
5.1.1 创建接口
packagestaticProxyPattern;// 创建法院接口publicinterfaceCourt{voiddoCourt();}
5.1.2 实现接口
packagestaticProxyPattern;publicclassPeopleimplementsCourt{//实现Court接口//在法庭上张三自己进行辩解@OverridepublicvoiddoCourt(){System.out.println("张三说自己没有偷东西");}}
5.1.3 对实现类添加功能
packagestaticProxyPattern;publicclassLawyerimplementsCourt{//接收需要代理的对象Courtzhangsan;publicLawyer(Courtzhangsan){this.zhangsan=zhangsan;}publicLawyer(){}//张三请律师进行辩解@OverridepublicvoiddoCourt(){//张三自己先辩解zhangsan.doCourt();//律师替张三辩解System.out.println("经视频证明,并不是当事人偷得东西");}}
5.1.4 测试
packagestaticProxyPattern;publicclassTest{publicstaticvoidmain(String[]args){//张三请律师打官司Peoplezhangsan=newPeople();Lawyerlawyer=newLawyer(zhangsan);lawyer.doCourt();}}
5.1.5 优缺点
优点:可以做到在符合开闭原则的情况下对目标对象进行功能扩展。
缺点:我们得为每一个服务都得创建代理类,工作量太大,不易管理;同时接口一旦发生改变,代理类也得相应修改。
如何解决静态代理的缺点呢? 这里就得说说动态代理了。
在动态代理中我们不再需要再手动的创建代理类,我们只需要编写一个动态处理器就可以了。真正的代理对象由JDK再运行时为我们动态的来创建。
5.2 JDK中生成代理对象的API
代理类所在包:java.lang.reflect.Proxy;
JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:
staticObjectnewProxyInstance(ClassLoaderloader,Class<?>[]interfaces,InvocationHandlerh)
返回值:Object就是代理对象
ClassLoaderloader//代表与目标对象相同的类加载器-------目标对象.getClass().getClassLoader()Class<?>[]interfaces//代表与目标对象实现的所有的接口字节码对象数组----数组因为目标类可以有多个接口InvocationHandlerh//事件处理,执行目标对象的方法时,会触发事件处理器的方法,//会把当前执行目标对象的方法作为参数传入。
5.3 动态代理代码实现
5.3.1 创建接口
packagestaticProxyPattern;// 创建法院接口publicinterfaceCourt{voiddoCourt();}
5.3.2 实现接口
packagestaticProxyPattern;publicclassPeopleimplementsCourt{//实现Court接口//在法庭上张三自己进行辩解@OverridepublicvoiddoCourt(){System.out.println("张三说自己没有偷东西");}}
5.3.3 代理律师
packagestaticProxyPattern;importjava.lang.reflect.InvocationHandler;importjava.lang.reflect.Method;importjava.lang.reflect.Proxy;publicclassLawyer{publicObjectgetLawyer(Courtzhangsan){returnProxy.newProxyInstance(//对象的类加载器zhangsan.getClass().getClassLoader(),//该对象实现的所有的接口newClass[]{Court.class},//对该对象添加功能newInvocationHandler(){//proxy就是目标对象,method就是调用目标对象中方法,args就是调用目标对象中方法的参数。//proxy的作用://1.可以使用反射获取代理对象的信息(也就是proxy.getClass().getName())。//2.可以将代理对象返回以进行连续调用,这就是proxy存在的目的,因为this并不是代理对象。@OverridepublicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{//律师进行申辩System.out.println("经视频证明,并不是当事人偷得东西");//张三自己进行申辩Objectinvoke=method.invoke(zhangsan,args);System.out.println("法院宣布,张三无罪释放");returninvoke;}});}}
5.3.4 测试代码
packagestaticProxyPattern;publicclassTest{publicstaticvoidmain(String[]args){//张三请律师打官司Peoplezhangsan=newPeople();Courtlawyer=(Court)newLawyer().getLawyer(zhangsan);lawyer.doCourt();}}
5.3.5 优缺点
优点:
相对于静态代理,动态代理大大减少了我们的开发任务,同时减少了对业务接口的依赖,降低了耦合度。
缺点:
代理对象不需要实现接口,但是目标对象一定要实现接口,否则不能用动态代理。
网友评论