代理模式

代理模式是常见设计模式的一种,代理模式的定义是:为其他对象提供一种代理以控制对这个对象的访问

在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。

静态代理

理解设计模式是比较枯燥的,所以还是以举例子的方式来进行理解,

例如:公司开年会想找个明星来表演,那么并不会直接联系明星(主要还是联系不上),而是会联系明星的经纪人,明星就是被代理的对象,而经纪人就是代理对象。明星只需要准备来参加年会时应该表演什么节目就可以,其他的出场费之类的事情就交给经纪人来处理就好了。代理对象可以理解为被代理对象的扩展,能做被代理对象不能做的事情,也可以调用代理对象做事情。

那么用代码实现这个场景是什么样子的呢?

执行合作方法的接口

/**
* @Description: 经纪公司接口,代理对象和被代理对象都需要实现的接口
*/
public interface Company {
/** 合作 */
void cooperation();
}

被代理对象

/**
* @Description: 目标对象-明星(被代理对象)
*/
public class Start implements Company { @Override
public void cooperation() {
System.out.println("is show time");
}
}

代理对象

/**
* @Description: 经纪人(代理对象)
*/
public class Agent implements Company { private Company company; public Agent(Company company)
{
this.company = company;
} @Override
public void cooperation()
{
System.out.println("收出场费,化妆等等");
company.cooperation();
System.out.println("收拾行李,打道回府");
}
}

测试类

import org.junit.Test;

/**
* @Description: 测试类
*/
public class ProxyTest { @Test
public void AnnualMeeting()
{
//目标对象
Start start = new Start();
//构建代理对象,生成代理关系
Agent agent = new Agent(start);
//用代理对象执行被代理对象的动作
agent.cooperation(); } }

输出结果:

收出场费,化妆等等
is show time
收拾行李,打道回府

静态代理的特点是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。有时候不方便修改别人的代码或者是引入的一个功能,需要进行功能扩展一下才能适用于自己的业务实现,可以使用代理模式来进行设计。

但是静态代理的实现基础是一个目标对象对应一个代理对象,并且在编译时就已经维护好了代理关系,如果目标对象是多个那么就会需要多个代理对象,这样在更新目标的对象的时候还需要更新代理对象,当代理对象持续增加时维护成本就变得非常困难。

针对于这种情况,动态代理应运而生。

动态代理

JDK代理

动态代理的代理对象不需要和目标对象共同实现接口,而是利用JDK的API,动态的在内存中构建代理对象。

动态生成代理对象需要调用JDK中的java.lang.reflect.Proxy类的newProxyInstance方法,这个方法需要三个参数:

@CallerSensitive
public static Object newProxyInstance(ClassLoader loader,]Class<?>[] interfaces,InvocationHandler h)
ClassLoader loader:类加载器,用来加载目标对象类,因为是在运行时获得目标对象,所以肯定需要用到反射。
Class<?>[] interfaces:目标对象类实现的接口集合,这些接口中定义目标对象可以执行的方法。
InvocationHandler h:这个参数代表的是动态代理对象在调用方法的时候,会将方法转发到哪一个invocationHandler对象身上,InvocationHandler是个接口,
需要自己实现它,然后定义自己的动态代理执行方法。
创建包含动态代理对象具体执行方法的实现类。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; /**
* @Description: 包含动态代理对象具体执行方法的实现类
*/
public class MyInvocationHandler implements InvocationHandler { private Company company; public MyInvocationHandler(Company company)
{
this.company = company;
} @Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
{ System.out.println("收出厂费,化妆等");
//具体执行方法
Object result = method.invoke(company,args); System.out.println("收拾现场,卸妆,打道回府"); return result;
}
}

测试类

import org.junit.Test;
import java.lang.reflect.Proxy; /**
* @Description: 测试类
*/
public class DynamicProxyTest { @Test
public void AnnualMeeting()
{
//创建目标对象
Company start = new Start();
//创建代理对象需要执行的方法处理对象
MyInvocationHandler myInvocationHandler = new MyInvocationHandler(start);
//获得目标对象的类加载器
ClassLoader classLoader = start.getClass().getClassLoader();
//创建动态代理对象
Company proxy = (Company) Proxy.newProxyInstance(classLoader,start.getClass().getInterfaces(),myInvocationHandler);
//用动态代理对象执行目标对象的方法
proxy.cooperation();
}
}

输出结果:

收出厂费,化妆等
is show time
收拾现场,卸妆,打道回府

JDK动态代理的特点:代理对象不需要实现接口,但是目标对象必须实现接口。

那么如果在实际的业务中目标对象确实没有实现接口,怎么办呢?

遇到这种情况的时候就需要时cglib动态代理了。

Cglib代理

Cglib代理,也叫作子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能的扩展.

  • JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口,如果想代理没有实现接口的类,就可以使用Cglib实现。
  • Cglib是一个强大的高性能的代码生成包,它可以在运行期扩展java类与实现java接口.它广泛的被许多AOP的框架使用,例如Spring AOP和synaop,为他们提供方法的interception(拦截)。
  • Cglib包的底层是通过使用一个小块的字节码处理框架ASM来转换字节码并生成新的类.不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。

在实现cglib代理时需要引入cglib的jar包,但是spring核心功能已经包含了cglib的功能,所以引入spring-core的jar包就可以了。

需要注意的是:代理的类不能为final,否则报错,目标对象的方法如果为final/static,那么就不会被拦截,即不会执行目标对象额外的业务方法。

没有实现接口的目标对象类

/**
* @Description: 没有经纪公司的明星,就行像最近以个人练习生出道的蔡徐坤
*/
public class AloneStart {
/** 合作 */
public void cooperation() {
System.out.println("is show time");
} }

生成Cglib代理对象的类

import org.mockito.cglib.proxy.Enhancer;
import org.mockito.cglib.proxy.MethodInterceptor;
import org.mockito.cglib.proxy.MethodProxy;
import java.lang.reflect.Method; /**
* @Description: 生成代理对象的类
*/
public class CglibProxy implements MethodInterceptor { private AloneStart aloneStart; public CglibProxy(AloneStart aloneStart)
{
this.aloneStart = aloneStart;
} /**
* 创建代理对象
* @return
*/
public Object getProxyInstance()
{
//动态代理工具类
Enhancer enhancer = new Enhancer();
//设置父类
enhancer.setSuperclass(aloneStart.getClass());
//设置回调函数调用对象
enhancer.setCallback(this);
//返回代理对象
return enhancer.create(); } @Override
public Object intercept(Object obj, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable
{
System.out.println("收出厂费,化妆等");
//执行代理方法
methodProxy.invokeSuper(obj,objects);
System.out.println("卸妆,回家");
return null;
}
}

测试类

import org.junit.Test;

/**
* @Description: 测试类
*/
public class CglibProxyTest { @Test
public void cglibTest()
{
//创建目标对象
AloneStart aloneStart = new AloneStart();
//创建代理对象
AloneStart startProxy = (AloneStart) new CglibProxy(aloneStart).getProxyInstance();
//用代理对象执行目标对象的方法
startProxy.cooperation();
}
}

输出结果:

收出厂费,化妆等
is show time
卸妆,回家

jdk采用反射机制调用委托类的方法,而cglib采用类似索引的方式直接调用委托类方法;

还有需要注意的是:

在Spring的AOP中

如果加入容器的目标对象有实现接口,用JDK代理
如果目标对象没有实现接口,用Cglib代理

参考:

Java的三种代理模式: https://www.cnblogs.com/cenyu/p/6289209.html

说说代理模式:http://www.importnew.com/26116.html

Java设计模式学习记录-代理模式的更多相关文章

  1. Java设计模式学习记录-模板方法模式

    前言 模板方法模式,定义一个操作中算法的骨架,而将一些步骤延迟到子类中.使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定步骤. 模板方法模式 概念介绍 模板方法模式,其实是很好理解的,具体 ...

  2. Java设计模式学习记录-状态模式

    前言 状态模式是一种行为模式,用于解决系统中复杂的对象状态转换以及各个状态下的封装等问题.状态模式是将一个对象的状态从该对象中分离出来,封装到专门的状态类中,使得对象的状态可以灵活多变.这样在客户端使 ...

  3. Java设计模式学习记录-备忘录模式

    前言 这次要介绍的是备忘录模式,也是行为模式的一种 .现在人们的智能手机上都会有备忘录这样一个功能,大家也都会用,就是为了记住某件事情,防止以后自己忘记了.那么备忘录模式又是什么样子的呢?是不是和手机 ...

  4. Java设计模式学习记录-迭代器模式

    前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...

  5. Java设计模式学习记录-解释器模式

    前言 这次介绍另一个行为模式,解释器模式,都说解释器模式用的少,其实只是我们在日常的开发中用的少,但是一些开源框架中还是能见到它的影子,例如:spring的spEL表达式在解析时就用到了解释器模式,以 ...

  6. Java设计模式学习记录-命令模式

    前言 这次要介绍的是命令模式,这也是一种行为型模式.最近反正没有面试机会我就写博客呗,该投的简历都投了.然后就继续看书,其实看书也会给自己带来成就感,原来以前不明白的东西,书上已经给彻底的介绍清楚了, ...

  7. Java设计模式学习记录-外观模式

    前言 这次要介绍的是外观模式(也称为门面模式),外观模式也属于结构型模式,其实外观模式还是非常好理解的,简单的来讲就是将多个复杂的业务封装成一个方法,在调用此方法时可以不必关系具体执行了哪些业务,而只 ...

  8. Java设计模式学习记录-桥接模式

    前言 这次介绍结构型设计模式中的第二种模式,桥接模式. 使用桥接模式的目的就是为了解耦,松散的耦合更利于扩展,但是会增加相应的代码量和设计难度. 桥接模式 桥接模式是为了将抽象化与实现化解耦,让二者可 ...

  9. Java设计模式学习记录-建造者模式

    前言 今天周末,有小雨,正好也不用出门了,那就在家学习吧,经过了两周的面试,拿到了几个offer,但是都不是自己很想去的那种,要么就是几个人的初创小公司,要么就是开发企业内部系统的这种传统开发,感觉这 ...

随机推荐

  1. php apache

    php与Apache配置基础知识 1.在 AllowOverride 设置为 None 时,.htaccess 文件将被完全忽略.当此指令设置为All时,所有具有“.htaccess” 作用域的指令都 ...

  2. 1.mysql安装

    Navicat账号:root 密码:weihu 账号:weihu 密码:weihu 1.首先进入的是安装引导界面 2.然后进入的是类型选择界面,这里有3个类型:Typical(典型).Complete ...

  3. C# webservice服务跟踪调试方法(转)

    1.新建网站,添加服务,并创建服务. 2.打开internet 信息服务管理器,添加网站,映射到创建的服务所在网站的目录. 3.打开服务所在网站的解决方案,进行配置. 1) 设置启动选项 选择启动操作 ...

  4. C# Winform模仿百度日历

    想写博客不知道从何处开始,就从回忆开始吧. 第一个就从自定义日历控件开始 产生背景: 大概2015年时候有个项目要用到日历,用默认日历展示给用户看,用户毫不客气都说界面太丑,最好做成像百度日历那样方便 ...

  5. C#操作数据表中XML格式的数据

    以前还真没有见过数据表中存储XML格式的数据,刚开始听说的时候,还以为是数据表中有XML的字段类型, 再了解,其实也就是字符串类型的,只不过字符串的格式是XML格式的.确实孤陋寡闻!汗... (可添加 ...

  6. 在windows10上创建ASP.NET mvc5+Memcached服务

    感谢两位两位大佬: https://blog.csdn.net/l1028386804/article/details/61417166 https://www.cnblogs.com/running ...

  7. Lerning Entity Framework 6 ------ Using a commandInterceptor

    Sometimes, We want to check the original sql statements. creating a commandInterceptor is a good way ...

  8. Apache-通过CGI执行脚本

    1.配置服务器,开启注释 vim /etc/httpd/conf/httpd.conf 292 # (You will also need to add "ExecCGI" to ...

  9. 关于scanf、getchar、getch、getche缓冲区分析——C语言

    缓冲区 根据数据刷新的时机可以将缓冲区的类型分为:全缓冲.行缓冲.无缓冲 (注意:Windows下的输出设备没有缓冲区,意思是printf是无缓冲的,但是在Linux下printf就是行缓冲的,至于为 ...

  10. 前端基础-html 字体标签,排版标签,超链接,图片标签

    主要内容: 字体标签: h1~h6.<font>.<u>.<b>.<strong><em>.<sup>.<sub> ...