java代理模式(jdk proxy)

目录
  • 什么是代理
    • 举个栗子
  • 什么是代理模式
  • 实现代理的方式
    • 静态代理
    • 根据以上过程,分析静态代理的优缺点
    • 动态代理
    • CGLIB代理
    • JDK代理
      • InvocationHandler接口
    • Method类
    • Proxy类
  • jdk动态代理的实现步骤
  • 总结

什么是代理

举个栗子

比如有一家美国的大学,可以对全世界招生。但是对于家长来说,家长不能直接自己去找学校,家长没有能力去直接访问学校,或者说,美国学校不接受个人来访,那么此时就需要一个留学中介来帮助这家美国学校招

生,中介就是学校的代理。中介和学校要做的事情是一致:招生。对于家长来说,学校就是目标,留学中介就是代理。日常生活中,有许多代理的例子,比如:代购,房产中介,各种中介,换ip,商家厂家和买家。在开发中

也有同样的情况,比如,你有a类, 本来是调用c类的方法, 完成某个功能。 但是c不让a调用。 a -----不能调用 c的方法。在a 和 c 直接 创建一个 b 代理, c让b访问。 a --访问b---访问c。

原来的访问关系

通过代理的访问关系

什么是代理模式

百度百科

代理模式是指,为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户类和目标对象之间起到中介的作用。

换句话说,使用代理对象,是为了在不修改目标对象的基础上,增强主业务逻辑。客户类真正的想要访问的对象是目标对象,但客户类真正可以访问的对象是代理对象。客户类对目标对象的访问是通过访问代理对象来实现

的。当然,代理类与目标类要实现同一个接口。

实现代理的方式

静态代理

静态代理是指,代理类在程序运行前就已经定义好.java 源文件,其与目标类的关系在程序运行前就已经确立。在程序运行前代理类已经编译为.class 文件。

举一个静态代理的例子

需求:用户需要购买 u 盘,u 盘厂家不单独接待零散购买,厂家规定一次最少购买 1000个以上,用户可以通过淘宝的代理商,或者微商哪里进行购买。淘宝上的商品,微商都是 u 盘工厂的代理商, 他们代理对 u 盘的销售业

务。用户购买-------代理商(淘宝,微商)----- u 盘厂家(金士顿,闪迪等不同的厂家)

1、定义业务接口

定义业务接口UsbSell(目标接口),其中含有抽象方法sell(int amout);sell是目标方法

public interface UsbSell {
    /**
     * 表示功能的,厂家和商家都要完成的功能
     * @param amount
     * @return
     */
    float sellUsb(int amount);
}

2、定义接口的实现类

目标类UsbKingFactory金士顿U盘,该类实现了接口

import school.xauat.service.UsbSell;
public class UsbKingFactory  implements UsbSell {
    @Override
    /**
     * 定义金士顿厂家的销售价格
     */
    public float sellUsb(int account) {
        return 75.0f;
    }
}

3、定义代理

TaoBao就是一个代理类,代理厂家销售U盘

import school.xauat.factory.UsbKingFactory;
import school.xauat.service.UsbSell;
public class Taobao implements UsbSell {
    //声明 商家代理的厂家具体是哪一家
    private UsbSell factory=new UsbKingFactory();
    @Override
    /**
     * 实现销售U盘的功能
     */
    public float sellUsb(int account) {
        float price=factory.sellUsb(account);
        //代理增强功能
        price+=25;
        return price;
    }
}

WeiShang也是一个代理类代理厂家销售U盘

import school.xauat.factory.UsbKingFactory;
import school.xauat.service.UsbSell;
public class Weishang implements UsbSell {
    //声明 商家代理的厂家具体是哪一家
    private UsbSell factory=new UsbKingFactory();
    @Override
    public float sellUsb(int amount) {
        float price=factory.sellUsb(amount);
        //代理增强功能
        price+=15;
        return price;
    }
}

4、客户端调用者,购买商品类

客户端可以通过Taobao和WeiShang两个代理来购买U盘

import school.xauat.business.Taobao;
import school.xauat.business.Weishang;
public class ShopMain {
    public static void main(String[] args) {
        Taobao taoBao=new Taobao();
        float price=taoBao.sellUsb(1);
        System.out.println(price);
        Weishang weishang=new Weishang();
        float price2=weishang.sellUsb(1);
        System.out.println(price2);
    }
}

根据以上过程,分析静态代理的优缺点

优点:实现简单,易于理解

缺点:

代码复杂,难于管理

代理类和目标类实现了相同的接口,每个代理都需要实现目标类的方法,这样就出现了大量的代码重复。如果接口增加一个方法,除了所有目标类需要实现这个方法外,所有代理类也需要实现此方法。增加了代码维护的复杂度。

代理类依赖目标类,代理类过多

代理类只服务于一种类型的目标类,如果要服务多个类型。势必要为每一种目标类都进行代理,静态代理在程序规模稍大时就无法胜任了,代理类数量过多。

动态代理

动态代理是指代理类对象在程序运行时由 JVM 根据反射机制动态生成的。动态代理不需要定义代理类的.java 源文件。

动态代理其实就是 jdk 运行期间,动态创建 class 字节码并加载到 JVM。

动态代理的实现方式常用的有两种:使用 JDK 动态代理(这里主要讲),与通过 CGLIB 动态代理

CGLIB代理

CGLIB(Code Generation Library)是一个开源项目。是一个强大的,高性能,高质量的 Code 生成类库,它可以在运行期扩展 Java 类与实现 Java 接口。它广泛的被许多 AOP 的框架使用,例如 Spring AOP。使用 JDK 的

Proxy 实现代理,要求目标类与代理类实现相同的接口。若目标类不存在接口,则无法使用该方式实现。但对于无接口的类,要为其创建动态代理,就要使用 CGLIB 来实现。CGLIB 代理的生成原理是生成目标类的子类,而

子类是增强过的,这个子类对象就是代理对象。所以,使用CGLIB 生成动态代理,要求目标类必须能够被继承,即不能是 final 的类。cglib 经常被应用在框架中,例如 Spring ,Hibernate 等。Cglib 的代理效率高于 Jdk。对

于 cglib 一般的开发中并不使用。做了一个了解就可以。

JDK代理

jdk 动态代理是基于 Java 的反射机制实现的。使用 jdk 中接口和类实现代理对象的动态创 建。 Jdk 的动态要求目标对象必须实现接口,这是 java 设计上的要求。 从 jdk1.3 以来,java 语言通过 java.lang.reflect 包提供三个类

支持代理模式 Proxy, Method InovcationHandler

InvocationHandler接口

InvocationHandler 接口叫做调用处理器,负责完调用目标方法,并增强功能。通 过 代 理 对 象 执 行 目 标 接 口 中 的 方 法 , 会 把 方 法 的 调 用 分 派 给 调 用 处 理 器(InvocationHandler)的实现类,执行实现类中的 i

nvoke()方法,我们需要把功能代理写在 invoke()方法中 。

在 invoke 方法中可以截取对目标方法的调用。在这里进行功能增强。Java 的动态代理是建立在反射机制之上的。实现了 InvocationHandler 接口的类用于加强目标类的主业务逻辑。这个接口中有一个方法 invoke(),具体加

强的代码逻辑就是定义在该方法中的。通过代理对象执行接口中的方法时,会自动调用 invoke()方法。

invoke()方法的介绍如下:

public Object invoke ( Object proxy, Method method, Object[] args)

proxy:代表生成的代理对象

method:代表目标方法

args:代表目标方法的参数

第一个参数 proxy 是 jdk 在运行时赋值的,在方法中直接使用,第二个参数后面介绍,第三个参数是方法执行的参数, 这三个参数都是 jdk 运行时赋值的,无需程序员给出。

Method类

invoke()方法的第二个参数为 Method 类对象,该类有一个方法也叫 invoke(),可以调用目标方法。这两个 invoke()方法,虽然同名,但无关。

public Object invoke ( Object obj, Object... args)

obj:表示目标对象

args:表示目标方法参数,就是其上一层 invoke 方法的第三个参数

该方法的作用是:调用执行 obj 对象所属类的方法,这个方法由其调用者 Method 对象确定。

在代码中,一般的写法为

method.invoke(target, args);

其中,method 为上一层 invoke 方法的第二个参数。这样,即可调用了目标类的目标方法。

Proxy类

通 过 JDK 的 java.lang.reflect.Proxy 类 实 现 动 态 代 理 , 会 使 用 其 静 态 方 法newProxyInstance(),依据目标对象、业务接口及调用处理器三者,自动生成一个动态代理对象。

public static newProxyInstance ( ClassLoader loader, Class<?>[] interfaces, InvocationHandler handler)

loader:目标类的类加载器,通过目标对象的反射可获取

interfaces:目标类实现的接口数组,通过目标对象的反射可获取

handler:调用处理器。

jdk动态代理的实现步骤

jdk 动态代理是代理模式的一种实现方式,其只能代理接口。

实现步骤

1、新建一个接口,作为目标接口

2、为接口创建一个实现类,是目标类

3、创建类实现 java.lang.reflect.InvocationHandler 接口,调用目标方法并增加其他功能代码

4、创建动态代理对象,使用 Proxy.newProxyInstance()方法,并把返回值强制转为接口类型。

举例

1、创建目标接口,定义目标接口功能

2、为接口创建实现类

以上两步同静态代理

3、创建InvocationHandler接口的实现类,调用目标方法和增加其他代码功能

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class MySellHandler implements InvocationHandler {
    //目标对象
    private Object target=null;
    public MySellHandler(Object target){
        this.target=target;
    }
    @Override
    /**
     * 实现InvocationHandler接口的实现类,在invoke方法中完成代理类的功能
     *      -调用目标方法
     *      -功能增强
     */
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result=null;
        result=method.invoke(target,args);

        //功能增强
        //这里为了简化,我们将功能增强定义为加价25元
        if (result!=null){
            float price=(float)result;
            price=price+25;
            result=price;
        }
        return result;
    }
}

这里的target对象相当于静态代理中的TaoBao和WeiShang

4、模拟客户购买U盘,使用proxy.newProxyInstance创建Proxy代理对象并且使返回值为目标接口类型

import school.xauat.factory.UsbKingFactory;
import school.xauat.handler.MySellHandler;
import school.xauat.service.UsbSell;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
public class MainShop {
    public static void main(String[] args) throws Exception {
        //创建目标对象
        Class c=UsbKingFactory.class;
        Object obj=c.newInstance();
        //获得目标类的类加载器
        ClassLoader loader =UsbKingFactory.class.getClassLoader();
        //获取目标类实现的接口数组
        Class<?>[]interfaces=obj.getClass().getInterfaces();
        //创建InvocationHandler对象
        InvocationHandler handler=new MySellHandler(obj);
        //创建代理对象
        UsbSell proxy=(UsbSell) Proxy.newProxyInstance(loader,interfaces,handler);
        //通过这个代理执行方法
        float price=proxy.sell(1);
        System.out.println(price);
    }
}

静态代理

动态代理

UML图

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • Java设计模式之代理模式详解

    一.代理模式 代理模式就是有一个张三,别人都没有办法找到他,只有他的秘书可以找到他.那其他人想和张三交互,只能通过他的秘书来进行转达交互.这个秘书就是代理者,他代理张三. 再看看另一个例子:卖房子 卖房子的步骤: 1.找买家 2.谈价钱 3.签合同 4.和房产局签订一些乱七八糟转让协议 一般卖家只在签合同的时候可能出面一下,其他的1,2,4都由中介去做.那你问这样有什么用呢? 首先,一个中介可以代理多个卖房子的卖家,其次,我们可以在不修改卖家的代码的情况下,给他实现房子加价.打广告等等夹带私货的

  • Java回调函数原理实例与代理模式的区别讲解

    java 回调函数例子,及其原理,与代理模式的区别 回调函数 应用程序(application program)会时常通过API调用库里所预先备好的函数.但是有些库函数(library function)却要求应用先传给它一个函数,好在合适的时候调用,以完成目标任务.这个被传入的.后又被调用的函数就称为回调函数(callback function). 作用: 可以把调用者与被调用者分开.调用者不关心谁是被调用者,所有它需知道的,只是存在一个具有某种特定原型.某些限制条件(如返回值为int)的被调

  • Java简单实现动态代理模式过程解析

    基本知识:JDK动态代理是java.lang.reflect.*包提供的方式,它必须借助一个接口才能生成代理对象. 步骤: 1.首先建一个接口类,并提供一个实现类: public interface ISubject { public int add(int a, int b); } public class Subject implements ISubject { @Override public int add(int a, int b) { return a + b; } } 2.创建一

  • Java线程创建静态代理模式代码实例

    一.背景 在通过Runnable接口创建线程时,启动线程需要借助Thread类,这里就涉及到了静态代理模式. 二.实例 以歌手演出为例,在演出的这个过程中,歌手与他的助理他们有一个共同的目标"完成这场演出". 为啥需要歌手需要有助理呢? 因为举办好一场演出有很多繁琐的事情要做,为了让歌手专心完成"唱歌"这件事,助理就需要在背后帮助歌手做很多事情. 1.助理负责帮助歌手做一些辅助工作,例如帮忙宣传.帮忙计划行程.帮忙订机票等等. 2.歌手负责唱歌这件事情. 三.实例的

  • java面试常见模式问题---代理模式

    本篇总结的是 代理设计模式,后续会经常更新~ 代理模式最直观的解释就是,通过代理,将被代理对象 "增强"!(即,扩展被代理对象的功能) 代理模式分为静态代理,和动态代理:动态代理的代理类是动态生成的 , 静态代理的代理类是我们提前写好的逻辑. Java 中实现动态代理的方式有 2 种: JDK 动态代理 CGLIB 动态代理 1.静态代理 静态代理角色分析: 抽象角色 :一般使用接口或者抽象类来实现. 真实角色 :被代理的角色. 代理角色: 代理真实角色 , 代理真实角色后 ,一般会做

  • Java代理模式实例详解【静态代理与动态代理】

    本文实例讲述了Java代理模式.分享给大家供大家参考,具体如下: 即Proxy Pattern,23种java常用设计模式之一.代理模式的定义:对其他对象提供一种代理以控制对这个对象的访问. Java的代理模式是Java中比较常用的设计模式,分为2中代理:静态代理与动态代理(JDK动态代理和cglib动态代理) 优点: 职责清晰 真实角色只需关注业务逻辑的实现,非业务逻辑部分,后期通过代理类完成即可. 高扩展性 不管真实角色如何变化,由于接口是固定的,代理类无需做任何改动. 缺点: 很明显的一点

  • java 代理模式及动态代理机制深入分析

    java 代理模式及动态代理机制深入分析 代理设计模式 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问.代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理. 代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 代理模式一般涉及到的角色有: 抽象角色:声明真实对象和代理对象的共同接口: 代理角色:代理对象角色内

  • 23种设计模式(7) java代理模式

    23种设计模式第七篇:java代理模式 定义: 为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用 角色: 1.抽象角色:声明真实对象和代理对象的共同接口. 2.代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象.同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装. 3.真实角色:

  • 一篇文章带你入门java代理模式

    目录 Java设计模式-代理模式 什么是代理模式? 代理模式 小结: 应用实例: 使用场景: 总结 Java设计模式-代理模式 什么是代理模式? 在代理模式(Proxy Pattern)中,一个类代表另一个类的功能.这种类型的设计模式属于结构型模式. 在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口. 所谓的代理严格来讲就是两个子类共同实现一个接口,其中一个子类负责真实业务实现,另一个辅助完成主类业务逻辑操作. 代理模式 UML图: 源代码: interface ISubject

  • Java代理模式的示例详解

    目录 定义 案例 需求 方案:静态代理模式 总结 定义 代理模式(Proxy Parttern) 为一个对象提供一个替身,来控制这个对象的访问,即通过代理对象来访问目标对象,这样做的话好处是可以在目标对象实现的基础上,进行额外的功能的扩展. 案例 需求 苹果公司通过苹果代理商来卖手机 方案:静态代理模式 定义抽象接口类,该类在代理模式中扮演的是一个抽象功能的角色,该案例中就是把出售手机抽象为了一个接口 /** * 售卖手机的接口(代理模式--抽象角色) * @author:liyajie * @

  • Java代理模式实例分析

    本文实例讲述了Java代理模式.分享给大家供大家参考,具体如下: 一 不用代理模式的酒商 1 代码 class RealSubject // 真实角色(红酒厂商) { public void sell() { System.out.println("我是红酒厂商,欢迎品尝购买"); } } public class NoProxytest { // 客户端 public static void main( String[] args ) { RealSubject sub = new

  • .Net结构型设计模式之代理模式(Proxy)

    目录 一.动机(Motivate) 二.意图(Intent) 三.结构图(Structure) 四.模式的组成 五.代理模式的分类: 六.代理模式的具体实现 七.代理模式的实现要点: 1.代理模式的优点: 2.代理模式的缺点: 3.代理模式的使用场景: 八..NET 中代理模式的实现 九.总结 一.动机(Motivate) 在面向对象系统中,有些对象由于某种原因(比如对象创建的开销很大,或者某些操作需要安全控制,或者需要进程外的访问等),直接访问会给使用者.或者系统结构带来很多麻烦.如何在不失去

  • java代理模式(jdk proxy)

    目录 什么是代理 举个栗子 什么是代理模式 实现代理的方式 静态代理 根据以上过程,分析静态代理的优缺点 动态代理 CGLIB代理 JDK代理 InvocationHandler接口 Method类 Proxy类 jdk动态代理的实现步骤 总结 什么是代理 举个栗子 比如有一家美国的大学,可以对全世界招生.但是对于家长来说,家长不能直接自己去找学校,家长没有能力去直接访问学校,或者说,美国学校不接受个人来访,那么此时就需要一个留学中介来帮助这家美国学校招 生,中介就是学校的代理.中介和学校要做的

  • Java设计模式之代理模式(Proxy模式)介绍

    理解并使用设计模式,能够培养我们良好的面向对象编程习惯,同时在实际应用中,可以如鱼得水,享受游刃有余的乐趣. Proxy是比较有用途的一种模式,而且变种较多,应用场合覆盖从小结构到整个系统的大结构,Proxy是代理的意思,我们也许有代理服务器等概念,代理概念可以解释为:在出发点到目的地之间有一道中间层,意为代理. 设计模式中定义:为其他对象提供一种代理以控制对这个对象的访问. 为什么要使用代理模式 1.授权机制 不同级别的用户对同一对象拥有不同的访问权利,如Jive论坛系统中,就使用Proxy进

  • java代理模式(静态代理、动态代理、cglib代理)

    目录 代理模式 静态代理 代码 接口 被代理对象 代理对象 测试 动态代理 代码: 接口 目标对象 代理对象 测试 cglib代理 代码: 目标对象 代理对象 测试 应用 总结 代理模式 代理模式(Proxy Pattern)是一种结构性模式.代理模式为一个对象提供了一个替身,以控制对这个对象的访问.即通过代理对象访问目标目标对象,可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能. 被代理的对象可以是远程对象.创建开销答得对象或需要安全控制得对象.代理模式主要有三种形式,分别

随机推荐