【Java入门提高篇】Day11 Java代理——JDK动态代理

  今天来看看Java的另一种代理方式——JDK动态代理

  我们之前所介绍的代理方式叫静态代理,也就是静态的生成代理对象,而动态代理则是在运行时创建代理对象。动态代理有更强大的拦截请求功能,因为可以获得类的运行时信息,可以根据运行时信息来获得更为强大的执(骚)行(操)力(作)。

  我们还是以上一个例子为例,这里的IStars接口和Stars类都不需要修改,只需要修改代理类。

  创建JDK动态代理需要先实现InvocationHandler接口,并重写其中的invoke方法,具体步骤如下:

  1. 创建一个类实现InvocationHandler接口。

  2. 给Proxy类提供委托类的ClassLoader和Interfaces来创建动态代理类。

  3. 利用反射机制得到动态代理类的构造函数。

  4. 利用动态代理类的构造函数创建动态代理类对象。

  我们用动态代理来改造一下之前的类:

  接口和委托类不需要修改:

public interface IStars {
    void sing();
    void dance();
}

  

public class Stars implements IStars{
    private String name;

    public Stars(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void sing(){
        System.out.println(getName() + " 唱了一首歌.");
    }

    public void dance(){
        System.out.println(getName() + " 跳了一支舞.");
    }
}

  这是使用动态代理后的代理类:

public class StarsNewProxy implements InvocationHandler {

    //代理类持有委托类的对象引用
    private Object object;

    //保存sing和dance的次数
    private int num;

    public StarsNewProxy(Object object){
        this.object = object;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if (!runBefore(method)){
            return null;
        };
        //利用反射机制将请求分派给委托类处理,Method的invoke返回Object对象作为方法执行结果
        Object result = method.invoke(object,args);
        runAfter(method);
        return result;
    }

    private boolean runBefore(Method method){
        System.out.println("我是代理,拦截到请求");
        if (method.getName().equals("dance")){
            System.out.println("抱歉,明星脚受伤了,不能跳舞表演了。");
            return false;
        }
        return true;
    }

    private void runAfter(Method method){
        System.out.println("我是代理,请求处理完毕");
    }
}

  新建一个工厂类来返回代理实例:

public class StarsNewProxyFactory {
    //构建工厂类,客户类调用此方法获得代理对象
    //对于客户类而言,代理类对象和委托类对象是一样的,不需要知道具体返回的类型
    public static IStars getInstance(String name){
        IStars stars = new Stars(name);
        InvocationHandler handler = new StarsNewProxy(stars);
        IStars proxy = null;
        proxy = (IStars) Proxy.newProxyInstance(
                stars.getClass().getClassLoader(),
                stars.getClass().getInterfaces(),
                handler
        );
        return proxy;
    }
}

  改写一下测试类:

public class Test {
    public static void main(String[] args){
//        testA();
        testB();
    }

    /**
     * 静态代理
     */
    private static void testA(){
        //创建目标对象
        IStars stars = new Stars("Frank");

        //代理对象,把目标传给代理对象,建立关系
        IStars starsProxy = new StarsProxy(stars);
        for (int i = 0;i < 5; i++){
            starsProxy.sing();
        }
    }

    /**
     * JDK动态代理
     */
    private static void testB(){
        IStars proxy = StarsNewProxyFactory.getInstance("Frank");
        proxy.dance();
        proxy.sing();
    }
}

  输出如下:

我是代理,拦截到请求
抱歉,明星脚受伤了,不能跳舞表演了。
我是代理,拦截到请求
Frank 唱了一首歌.
我是代理,请求处理完毕

  使用动态代理时实现了InvocationHandler接口并重写了invoke方法,invoke方法的三个参数:

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

proxy:  被代理的对象
method:  被代理对象的某个方法的Method对象
args:  被代理对象的某个方法接受的参数

  Proxy的newProxyInstance方法详情如下:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

loader:  一个ClassLoader对象,定义了由哪个ClassLoader对象来对生成的代理对象进行加载

interfaces:  一个Interface对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了

h:  一个InvocationHandler对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个InvocationHandler对象上

  可以看到,这里的动态代理跟静态代理一样,在代理类内部保存了一个委托类的实例,实际上都是调用原来的委托实例来进行需要的操作,代理类相当于给委托类加上一个外壳,把委托类置于代理类的内部,从而可以控制客户类对委托类的访问,就像上例中,代理类拦截了客户类对Stars类的dance方法的访问,并且输出了补充信息。

  动态代理跟静态代理最大的不同便是生成代理类的时期不同,静态代理是在编译期,而动态代理则是在运行时根据委托类信息动态生成。

  其次,动态代理实现的是InvocationHandler接口,而静态代理则是直接实现公共接口。当然动态代理也是需要实现相同的接口的,只是将接口信息放在了getInstance内部,相当于代理类跟委托类之间的约定,“这几个方法帮我代理一下吧”。

  最后,动态代理可以获得更多的运行时信息,使用起来也会更加灵活。

  至此,JDK动态代理讲解完毕,欢迎大家继续关注!

原文地址:https://www.cnblogs.com/mfrank/p/8116789.html

时间: 2024-11-29 09:22:55

【Java入门提高篇】Day11 Java代理——JDK动态代理的相关文章

【Java入门提高篇】Day12 Java代理——Cglib动态代理

今天来介绍另一种更为强大的代理--Cglib动态代理. 什么是Cglib动态代理? 我们先回顾一下上一篇的jdk动态代理,jdk动态代理是通过接口来在运行时动态创建委托类的代理对象,但是跟静态代理一样有一个缺点,就是必须和委托类实现相同的接口,当接口数量增加时,便需要增加代理类的数量才能满足需求,而且如果委托类是别人写的,而且没有实现任何接口,那么jdk动态代理就有些力不从心了. 这时候Cglib动态代理就脱颖而出了,Cglib并不依赖接口,可以直接生成委托类的代理对象,而且可以代理委托类的任意

java 静态代理 JDK动态代理 Cglib动态代理

下面以一个简单的银行账户为例讲述讲述动态代理. 设计一个银行账户类,包含用户的账户余额,实现查询和更新余额功能 这个系统用了一段时间,有客户要求对账说账户余额给弄错了?因为上面没有存取款记录,最后银行不认账,客户收到了损失.银行为了避免这种现象再次发生,决定对这个系统进行修改,但是因为bankAccount太过复杂,希望在不修改bankAccount的情况下,增加日志功能. 静态代理 使用静态代理解决上面的问题. 银行要求所有模块都需要添加日志功能,这对苦逼的程序员来说真的是一个不小的工作量啊,

Java代理之(jdk静态代理/jdk动态代理/cglib动态代理/aop/aspectj)

一.概念 代理是什么呢?举个例子,一个公司是卖摄像头的,但公司不直接跟用户打交道,而是通过代理商跟用户打交道.如果:公司接口中有一个卖产品的方法,那么公司需要实现这个方法,而代理商也必须实现这个方法.如果公司卖多少钱,代理商也卖多少钱,那么代理商就赚不了钱.所以代理商在调用公司的卖方法后,加上自己的利润然后再把产品卖给客户.而客户部直接跟公司打交道,或者客户根本不知道公司的存在,然而客户最终却买到了产品. 专业点说:代理模式是对象的结构型模式,代码模式给某一个对象提供代理,并由代理对象控制原对象

zbb20180930 代理模式 -静态代理-jdk动态代理-cglib动态代理

CGLIB与JDK动态代理区别 区别: java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理.而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理. 1.如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 2.如果目标对象实现了接口,可以强制使用CGLIB实现AOP 3.如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB

【Java入门提高篇】Day10 Java代理——静态代理

今天要介绍的是一个Java中一个很重要的概念--代理. 什么是代理?联系生活想想看,代理似乎并不陌生,最形象的代表便是经纪人,明星一般都有经纪人,经纪人作为中间人,负责代理明星的相关事宜,比如说,有人要请明星去唱歌表演,一般不会直接跟明星联系,而是联系他的经纪人,他的经纪人来负责安排行程,而真正唱歌表演的还是明星本人,经纪人仅仅作为一个附加物存在. 在Java中,代理也是这样的概念,来看个栗子: 先来创建一个明星类Stars: public class Stars implements ISta

【Java入门提高篇】Day5 Java中的回调(二)

Java中有很多个Timer,常用的有两个Timer类,一个java.util包下的Timer,一个是javax.swing包下的Timer,两个Timer类都有用到回调机制.可以使用它在到达指定时间间隔后发出通知,例如程序中有一个时钟,就可以每秒请求发送一个通知,以便更新时间显示.那么怎样通知Timer去完成我们的任务呢?在其他语言中,可以提供一个函数名,然后定时器周期性的调用它,但Java使用的是面向对象编程的思想,将某个类的对象传递给定时器,然后定时器再调用这个对象的方法,由于对象能携带的

【Java入门提高篇】Day16 Java异常处理(上)

当当当当当当,各位看官,好久不见,甚是想念. 今天我们来聊聊Java里的一个小妖精,那就是异常. 什么是异常?什么是异常处理? 异常嘛,顾名思义就是不正常,(逃),是Java程序运行时,发生的预料之外的事情,它阻止了程序按照程序员的预期正常执行. 异常处理,应该说异常处理机制,就是专门用来制服这个小妖精的法宝.Java中的异常处理机制能让程序在异常发生时,按照代码的预先设定的异常处理逻辑,针对性地处理异常,让程序尽最大可能恢复正常并继续执行,且保持代码的清晰. 简而言之,Java异常处理就是能让

【Java入门提高篇】Day15 Java泛型再探——泛型通配符及上下边界

上篇文章中介绍了泛型是什么,为什么要使用泛型以及如何使用泛型,相信大家对泛型有了一个基本的了解,本篇将继续讲解泛型的使用,让你对泛型有一个更好的掌握和更深入的认识. 上篇中介绍完泛型之后,是不是觉得泛型挺好用的?既消除了Object的不安全类型转化,又可以很方便的进行类型对象的存取,但是,等一下,有没有考虑到这样的情况. 我们先定义一个水果类: public class Fruit { private String name; public Fruit(String name){ this.na

模式的秘密-代理模式(2)-JDK动态代理

代理模式-动态代理 (1) (2) 代码实践动态代理: 第一步:被代理类的接口: package com.JdkProxy; public interface Moveable { void move(); } 第二步:被代理类: package com.JdkProxy; import java.util.Random; public class Car implements Moveable { @Override public void move() { //实现开车 try { Thre