代理模式-动态代理

(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 {
Thread.sleep(new Random().nextInt(1000));
System.out.println("汽车行驶中....");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

第三步:代理类:实现接口:InvocationHandler,同时把被代理类对象接口传入构造方法,

重写的接口的invoke方法。

package com.JdkProxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; public class TimeHandler implements InvocationHandler { private Object target; public TimeHandler(Object target)
{
this.target=target;
} /*
* 参数:
* proxy:被代理对象
* method:被代理对象方法
* arg:方法的参数
* 返回值:
* Object 方法的返回值
* */
@Override
public Object invoke(Object proxy, Method method, Object[] arg) throws Throwable { long starttime=System.currentTimeMillis();
System.out.println("汽车开始形式...."); method.invoke(target); long endtime=System.currentTimeMillis();
System.out.println("汽车结束行驶...汽车形式时间:"+(endtime-starttime)+"毫秒"); return null;
} }

测试类中实现代理:

使用Proxy类的newProxyInstance方法产生一个被代理类的实例,该实例可以当作被代理类使用接口(对应cls.getInterfaces())中声明过的方法。

package com.JdkProxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy; public class Test { /*JDK动态代理测试类
* */
public static void main(String[] args) { Car car=new Car(); //InvocationHandler是一个接口,接口中定义了一个方法invoke。要想实现JDK动态代理,
//代理类必须继承这个接口
InvocationHandler h=new TimeHandler(car);//
Class cls=car.getClass();//获取类对象,以便获取类加载器,以及获取类的接口 /*
* newProxyInstance返回代理类的实例,返回后的代理类可以当作被代理类使用
* (可使用被代理类的接口中声明过的方法)
* loader:类加载器
* interfaces:实现接口
* h:InvocationHandler
* */
Moveable m=(Moveable) Proxy.newProxyInstance(cls.getClassLoader(),
cls.getInterfaces(), h);

m.move(); } }

测试结果:

汽车开始形式....
汽车行驶中....
汽车结束行驶...汽车形式时间:863毫秒

代理模式-动态代理

所以动态代理是这样一种Class:

  • 他在运行时候产生了的Class
  • 该class需要实现一组interface
  • 使用动态代理类时,必须实现InvocationHandler接口

动态代理实现步骤

1,创建一个实现InvocationHandler的类,他必须实现Invoke方法

2,创建被代理的类以及接口

3,调用Proxy的静态方法,创建一个代理类:

newProxyInstance(ClassLoader,class[] interfaces,InvocationHandler  h)。

4,通过代理调用方法。

jdk动态代理只能实现了接口的类。

jdk动态代理理解:

看完这个例子,我的最大疑惑是,返回代理类实例,执行方法的时候,调用的是被代理类中含有的方法,那么我们代理类中重写的invoke方法,并没有调用,

那这个方法有什么用,但是真正执行被代理类里面的方法的时候,却好像又确实执行了invoke方法,那么到底时候用到了invoke这个方法呢

这里看了一下这篇博客的解释,稍微能理解一下这种动态代理的处理执行过程:https://juejin.im/post/5a99048a6fb9a028d5668e62。

在静态代理部分,我们在代理类中传入了被代理对象。可是,使用newProxyInstance生成动态代理对象的时候,我们居然不再需要传入被代理对象了
我们传入了的实际对象是InvocationHandler实现类的实例,这看起来有点像生成了InvocationHandler的代理对象,
在动态生成的代理类的任意方法中都会间接调用InvocationHandler->invoke(proxy, method, args)方法。 其实的确是这样。TimeProxy真正代理的对象就是InvocationHandler,不过这里设计的巧妙之处在于,InvocationHandler是一个接口,
真正的实现由用户指定。另外,在每一个方法执行的时候,invoke方法都会被调用 ,这个时候如果你需要对某个方法进行自定义逻辑处理,
可以根据method的特征信息进行判断分别处理

看完这段文字,再看这个方法的执行:

        Car car=new Car();
InvocationHandler h=new TimeHandler(car);
Class cls=car.getClass();
Moveable m=(Moveable) Proxy.newProxyInstance(cls.getClassLoader(),
cls.getInterfaces(), h);

总结步骤:

  • InvocationHandler:这个接口主要用于自定义代理逻辑处理
  • 为了完成对被代理对象的方法拦截,我们需要在InvocationHandler对象中传入被代理对象实例。
  • Proxy->newProxyInstance用于生成代理对象。
  • newProxyInstance的三个参数:
  • 前两个参数与被代理类的Class有关(被代理类的类加载器,被代理类的接口)。
  • 最后一个参数传入代理类InvocationHandler的实例。

想象一下,到此为止,如果我们还需要对其它任意对象进行代理,是否还需要改动newProxyInstance方法的源码,答案是:完全不需要!

只要你在newProxyInstance方法中指定代理需要实现的接口指定用于自定义处理的InvocationHandler对象

整个代理的逻辑处理都在你自定义的InvocationHandler实现类中进行处理。

至此,而我们终于可以从不断地写代理类用于实现自定义逻辑的重复工作中解放出来了,从此需要做什么,交给InvocationHandler。

答疑解惑

invoke方法的第一个参数proxy到底有什么作用?

1. 可以使用反射获取代理对象的信息(也就是proxy.getClass().getName(),输出可得:com.sun.proxy.$Proxy0)。
2. 可以将代理对象返回以进行连续调用,这就是proxy存在的目的。因为this并不是代理对象,指的是被代理对象实例(如例子中的car),

proxy指的是被代理对象。这个问题其实也好理解,如果你的接口中有方法需要返回自身,如果在invoke中没有传入这个参数,将导致实例无法正常返回。

在这种场景中,proxy的用途就表现出来了。简单来说,这其实就是最近非常火的链式编程的一种应用实现。

动态代理到底有什么用?

学习任何一门技术,一定要问一问自己,这到底有什么用。其实,在这篇文章的讲解过程中,我们已经说出了它的主要用途。你发现没,使用动态代理我们居然可以在不改变源码的情况下,直接在方法中插入自定义逻辑。这有点不太符合我们的一条线走到底的编程逻辑,这种编程模型有一个专业名称叫 AOP。所谓的AOP,就像刀一样,抓住时机,趁机插入。

 
 

模式的秘密-代理模式(2)-JDK动态代理的更多相关文章

  1. 浅谈Java代理一:JDK动态代理-Proxy.newProxyInstance

    浅谈Java代理一:JDK动态代理-Proxy.newProxyInstance java.lang.reflect.Proxy:该类用于动态生成代理类,只需传入目标接口.目标接口的类加载器以及Inv ...

  2. Spring AOP高级——源码实现(3)AopProxy代理对象之JDK动态代理的创建过程

    spring-aop-4.3.7.RELEASE  在<Spring AOP高级——源码实现(1)动态代理技术>中介绍了两种动态代理技术,当然在Spring AOP中代理对象的生成也是运用 ...

  3. 代理模式(静态代理、JDK动态代理原理分析、CGLIB动态代理)

    代理模式 代理模式是设计模式之一,为一个对象提供一个替身或者占位符以控制对这个对象的访问,它给目标对象提供一个代理对象,由代理对象控制对目标对象的访问. 那么为什么要使用代理模式呢? 1.隔离,客户端 ...

  4. 从静态代理,jdk动态代理到cglib动态代理-一文搞懂代理模式

    从代理模式到动态代理 代理模式是一种理论上非常简单,但是各种地方的实现往往却非常复杂.本文将从代理模式的基本概念出发,探讨代理模式在java领域的应用与实现.读完本文你将get到以下几点: 为什么需要 ...

  5. 代理模式详解:静态代理、JDK动态代理与Cglib动态代理

    代理模式简介分类 概念 ​ 代理,是为了在不修改目标对象的基础上,增强目标方法的业务逻辑. ​ 客户类需要执行的是目标对象的目标方法,但是真正执行的是代理对象的代理方法,客户类对目标对象的访问是通过代 ...

  6. 动态代理模式——JDK动态代理

    今天,我就来讲一下动态代理的设计模式. 动态代理的意义在于生成一个代理对象,来代理真实对象,从而控制真实对象的访问.操作动态代理需要两个步骤:一.代理对象和真实对象建立代理关系.二.实现代理对象的代理 ...

  7. Cglib 与 JDK动态代理的运行性能比较

    都说 Cglib 创建的动态代理的运行性能比 JDK 动态代理能高出大概 10 倍,今日抱着怀疑精神验证了一下,发现情况有所不同,遂贴出实验结果,以供参考和讨论. 代码很简单,首先,定义一个 Test ...

  8. java学习笔记(中级篇)—JDK动态代理

    一.什么是代理模式 相信大家都知道代理商这个概念,在商业中,代理商无处不在.假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商.你要买或 ...

  9. CGlib和JDK动态代理

    一.CGlib动态代理     JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了.CGLib采用了非常底层的1:字节码技术,其原理是通过字节 ...

  10. JDK动态代理、CGLIB动态代理详解

    Spring的AOP其就是通过动态代理的机制实现的,所以理解动态代理尤其重要. 动态代理比静态代理的好处: 1.一个动态代理类可以实现多个业务接口.静态代理的一个代理类只能对一个业务接口的实现类进行包 ...

随机推荐

  1. ZOJ 3707 Calculate Prime S 数论

    思路:容易得到s[n]=s[n-1]+s[n-2],也就是fib数. 求第k小的fib质数的也就是第k个质数数-2,当k>2时. 在就是s[n]/x%m=s[n]%(x*m)/x. 代码如下: ...

  2. tarjan算法--cojs 1298. 通讯问题

    cojs 1298. 通讯问题 ★   输入文件:jdltt.in   输出文件:jdltt.out   简单对比时间限制:1 s   内存限制:128 MB [题目描述] 一个篮球队有n个篮球队员, ...

  3. PIL The _imaging C module is not installed

    今天在WIN 7 64位用PIL的时候,提示 The _imaging C module is not installed ,原来是需要安装64位的. 刚开始安装的是这个:http://www.pyt ...

  4. 【原创】LogCat信息演示Activity生命周期

    界面如下:注意:这是在手机.竖屏状态下! (一)1个Activity     /**     * 7个方法     * 测试1个Activity的生命周期     *      * LogCat:   ...

  5. VMware中网络设置之host-only

    有了前面一篇的NAT的网络设置,本文就显得非常简单了.同样图文结合的步骤: 1.设置host-only模式. 2.设置linux虚拟机的静态IP.进入linux系统,点击主菜单---系统设置---网络 ...

  6. svn服务器搭建及使用 二

    上一篇介绍了VisualSVN Server和TortoiseSVN的下载,安装,汉化.这篇介绍一下如何使用VisualSVN Server建立版本库,以及TortoiseSVN的使用. 首先打开Vi ...

  7. TMapTextfile v.99/1

    By Hellinger Software. Class to handle text files as memory mapped files. Including efficient method ...

  8. 让linux history命令显示命令的运行时间、在哪个机器运行的这个命令

    1.在/etc/profile的最后加入例如以下部分: USER_IP=`who -u am i 2>/dev/null| awk '{print $NF}'|sed -e 's/[()]//g ...

  9. 无法执行磁盘检查因为windows无法访问

    固态硬盘迁移系统后,将原先的系统盘(H,重新系统后固态硬盘的主分区变为了C盘)格式化,并和邻近的D盘合并.合并后,将盘符重新改为D盘,而D盘原有的软件安装目录Program Files无法识别,显示目 ...

  10. 如何更改postgresql的最大连接数

    改文件 postgresql.conf 里的 #max_connections=32 为 max_connections=1024 以及另外相应修改 share_buffer 参数.