简介

什么是代理模式?

代理模式就是多一个代理类出来,代替原对象进行一些操作。比如说租房的中介、打官司的律师、旅行社,他们可以代替我们做一些事情,这就是代理。

代理模式的应用场景:

如果已有的方法在使用的时候需要对原有的方法进行改进,此时有两种办法:
1. 修改原有的方法来做到改进。但这样违反了“对扩展开放,对修改关闭”的原则。
2. 采用一个代理类调用原有的方法,且对产生的结果进行控制。这就是代理模式。

代理模式的分类:

  静态代理:由程序员创建代理类或特定工具自动生成源代码再对其编译。再程序运行前代理类的.class文件就已经存在了。

  动态代理:在程序运行时用反射机制,动态创建代理类。

静态代理

所谓静态就是在程序运行前就已经存在代理类的字节码文件,代理类和委托类的关系在运行前就确定了。

package Proxy;
//Created by zhengbinMac on 2017/2/19.
public interface Hello {
public void sayHello(String way);
}
package Proxy;
//Created by zhengbinMac on 2017/2/19.
public class HelloImpl implements Hello{
public void sayHello(String way) {
System.out.println("Hello by " + way);
}
}
package Proxy;
//Created by zhengbinMac on 2017/2/19.
public class HelloProxy implements Hello {
private Hello hello;
public HelloProxy() {
hello = new HelloImpl();
}
private void sayBefore() { System.out.println("Before.."); }
private void sayAfter() { System.out.println("After.."); }
public void sayHello(String way) {
sayBefore();
hello.sayHello(way);
sayAfter();
}
}

代理类和委托类要实现相同的接口,因为代理真正调用的还是委托类的方法。HelloProxy 类实现了 Hello 接口,在构造方法中 new 出一个 HelloImpl 类的实例。在 sayHello()方法中调用 HelloImpl 的 sayHello()方法,同时在调用的前后加上 before 与 after 方法。

测试类:

package Proxy;
//Created by zhengbinMac on 2017/2/19.
public class Test {
public static void main(String[] args) {
Hello proxy = new HelloProxy();
proxy.sayHello("normal");
}
}

输出结果:

Before..
Hello by normal
After..

以上就是静态代理,下面分析其带来的优缺点:

优点:

  客户端不必知道实现类(委托类)如何如何,只需要调用代理类即可。

缺点:

  • 代理类和委托类实现了相同的接口,代理类通过委托类实现了相同的方法。但这样出现了大量的代码重复。如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也要实现这个方法。这显然增加了代码的复杂度。
  • 代理对象只服务于一种类型的对象,如果要服务多类型的对象,那就要对每种对象都进行代理。静态代理子啊程序规模稍大是就无法胜任了。

JDK代理——接口级别代理

package Proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
//Created by zhengbinMac on 2017/2/19.
public class DynamicProxy implements InvocationHandler{
private Object target;
public DynamicProxy(Object target) { this.target = target; }
@SuppressWarnings("unchecked")
public <T> T getProxy() {
return (T) Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
this
);
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
sayBefore();
Object result = method.invoke(target, args);
sayAfter();
return result;
}
private void sayBefore() { System.out.println("before..."); }
private void sayAfter() { System.out.println("after..."); }
}

JDK 提供的 Proxy 类的工厂方法 newProxyInstance 去动态地创建一个 Hello 接口的代理类。

Proxy.newProxyInstance:

参数:

  • loader - 定义代理类的类加载器
  • interfaces - 代理类要实现的接口列表
  • h - 指派方法调用的调用处理程序(每个代理实例都具有一个关联的调用处理程序,调用代理实例的方法时,将对方法的调用指派到它的调用处理程序的 invoke 方法)

返回:

一个带有代理类的指定调用处理程序的代理实例,它由指定类加载器定义,并实现指定的接口。

测试类:

package Proxy;
//Created by zhengbinMac on 2017/2/19.
public class Test {
public static void main(String[] args) {
DynamicProxy dynamicProxy = new DynamicProxy(new HelloImpl());
Hello hello = dynamicProxy.getProxy();
hello.sayHello("JDK");
}
}

动态代理相比静态代理,接口变了,动态代理类不需要改变,而静态代理类不仅需要改变实现类,代理类也需要修改。

但如果要代理一个没有接口的类,JDK 动态代理就用不上了,这就引出了 CGLib 代理。

CGLib代理——方法级别代理

Spring、Hibernate 框架都是用了它,它是一个在运行期间动态生成字节码的工具,也就是动态生成代理类。

package Proxy;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
//Created by zhengbinMac on 2017/2/19.
public class CGLibProxy implements MethodInterceptor {
// 单例模式
private static CGLibProxy instance = new CGLibProxy();
private CGLibProxy() {}
public static CGLibProxy getInstance () { return instance; }
public <T> T getProxy(Class<T> cls) {
return (T) Enhancer.create(cls, this);
}
public Object intercept(Object obj, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
sayBefore();
Object result = methodProxy.invokeSuper(obj, objects);
sayAfter();
return result;
}
private void sayBefore() { System.out.println("before..."); }
private void sayAfter() { System.out.println("after..."); }
}

CGLib 给我们提供的是方法级别的代理,也可以理解为对方法的拦截。通过直接调用 proxy 的 invokeSuper 方法,将被代理的对象 obj 以及方法的参数 objects 传入其中即可。

测试类:

package Proxy;
//Created by zhengbinMac on 2017/2/19.
public class Test {
public static void main(String[] args) {
Hello helloCGLib = CGLibProxy.getInstance().getProxy(HelloImpl.class);
helloCGLib.sayHello("CGLib");
}
}

设计模式——代理模式(静态代理和JDK、CGLib动态代理)的更多相关文章

  1. java代理(静态代理和jdk动态代理以及cglib代理)

    版权声明:本文为Fighter168原创文章,未经允许不得转载.   目录(?)[+]   说到代理,脑袋中浮现一大堆代理相关的名词,代理模式,静态代理,jdk代理,cglib代理等等. 记忆特别深刻 ...

  2. 代理模式详解:静态代理+JDK/CGLIB 动态代理实战

    1. 代理模式 代理模式是一种比较好的理解的设计模式.简单来说就是 我们使用代理对象来代替对真实对象(real object)的访问,这样就可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标 ...

  3. jdk动态代理和cglib动态代理底层实现原理详细解析(cglib动态代理篇)

    代理模式是一种很常见的模式,本文主要分析cglib动态代理的过程 1. 举例 使用cglib代理需要引入两个包,maven的话包引入如下 <!-- https://mvnrepository.c ...

  4. Java设计模式之代理模式(静态代理和JDK、CGLib动态代理)以及应用场景

    我做了个例子 ,需要可以下载源码:代理模式 1.前言: Spring 的AOP 面向切面编程,是通过动态代理实现的, 由两部分组成:(a) 如果有接口的话 通过 JDK 接口级别的代理 (b) 如果没 ...

  5. 静态代理和jdk动态代理

    要说动态代理,必须先聊聊静态代理. 静态代理 假设现在项目经理有一个需求:在项目现有所有类的方法前后打印日志. 你如何在不修改已有代码的前提下,完成这个需求? 我首先想到的是静态代理.具体做法是: 1 ...

  6. 动态代理:JDK原生动态代理(Java Proxy)和CGLIB动态代理原理+附静态态代理

    本文只是对原文的梳理总结,以及自行理解.自己总结的比较简单,而且不深入,不如直接看原文.不过自己梳理一遍更有助于理解. 详细可参考原文:http://www.cnblogs.com/Carpenter ...

  7. 基于SpringBoot实现AOP+jdk/CGlib动态代理详解

    动态代理是一种设计模式.在Spring中,有俩种方式可以实现动态代理--JDK动态代理和CGLIB动态代理. JDK动态代理 首先定义一个人的接口: public interface Person { ...

  8. java静态代理和JDK动态代理

    静态代理 编译阶段就生产了对应的代理类 public interface IBussiness { void execute(); } public class BussinessImpl imple ...

  9. Spring Boot @Trasactionl 失效, JDK,CGLIB动态代理

    来自: https://www.cnblogs.com/sweetchildomine/p/6978037.html?utm_source=itdadao&utm_medium=referra ...

随机推荐

  1. golang学习笔记 ---命令行参数

    os 包以跨平台的方式,提供了一些与操作系统交互的函数和变量.程序的命令行参数可从os包的Args变量获取:os包外部使用os.Args访问该变量. os.Args变量是一个字符串(string)的切 ...

  2. Smarty标签运算,控制结构[if,for,foreach,section,while]

    Smarty标签运算: 在页面上做简单的运算[temp5.html] 条件判断if 循环结构 for foreach用得比较多,foreach例子从数据库取出的数据 section功能和foreach ...

  3. 对于不返回任何键列信息的 selectcommand 不支持 updatecommand 的动态 sql 生成

    大家知道,DataSet保存的数据是位于服务器内存里面的原数据库的“副本”.所以用DataSet更新数据的过程就是先对“副本”进行更新,然后 在将“原本”更新,按照我的理解就是把“原本”覆盖掉.具体到 ...

  4. Oracle 12C -- ADRCI查看DDL日志

    $ adrci ADRCI: Release - Production on Tue Nov :: Copyright (c) , , Oracle and/or its affiliates. Al ...

  5. the most beautiful media player on the linux platform.

    the most beautiful media player on the linux platform------> deepin media player http://wiki.linu ...

  6. MFC中无标题栏窗口的移动

    原文链接: http://blog.sina.com.cn/s/blog_6288219501015dwa.html   移动标准窗口是通过用鼠标单击窗口标题条来实现的,但对于没有标题条的窗口,就需要 ...

  7. x电容和Y电容

    https://wenku.baidu.com/view/c0a68a6a4a7302768e9939bd.html 根據 IEC 60384-14, 電容器分為 X 電容及 Y 電容 , 1. X ...

  8. IAR注释的快捷键

    1.注释的快捷键:Ctrl+K;取消注释:Ctrl+Shfit+K ... IAR注释代码的时候和MDK有很大的不同,以下简要说之:在注释代码的时候,你需如此操作: 注释一行,直接使用双斜杠”//“即 ...

  9. struts2:多模块多配置文件开发

    struts2支持多模块多配置文件开发.下面是一个仅包含两个模块的示范程序,包括财务.仓库模块.它们都有一个“caiwu“的Action,在各自的命名空间下:还有一个从财务转向到仓库的Action. ...

  10. 使用gradle多渠道打包

    以友盟的多渠道打包为例,如果我们须要打包出例如以下渠道:UMENG, WANDOUJIA, YINGYONGBAO. 第一种方法.是须要创建文件的. 我们在写完我们的代码之后,在app/src以下.分 ...