一旦这样绑定后,那么在进入代理对象方法调用的时候就会到HelloServiceProxy的invoke方法上,invoke方法有三个参数:第一个proxy是代理对象,第二个是当前调用那个方法,第三个是方法的参数。

2、JDK动态代理

这个动态代理还是不好理解的,我们先用一张图来表达它:

好,它的意义是在我们访问一个对象的时候,动态代理可以给我们生成一个代理对象【占位】,它的作用是可以控制【真实对象】的访问。

比方说,你是未成年人,我可能是是去访问你父母,由你父母来代替你访问你,那么你父母就是这个占位,而你就是真实对象。

在JDK动态代理的要求是对象存在接口,所以真正的服务对象必须存在接口。

假设我们现在有这样的一个服务,我给名字它会说: “hello ”+名字。

那么我们先看看这个接口:

  1. public interface HelloService {
  2. public void sayHello(String name);
  3. }

非常简单的接口,然后我们给出实现类:

  1. public class HelloServiceImpl implements HelloService{
  2. @Override
  3. public void sayHello(String name) {
  4. System.err.println("hello " + name);
  5. }
  6. }

好,它实现了这个接口,并且实现了接口的方法。

现在我们要变变,我希望

在说:hello+name前打印:我准备说hello。

之后打印:我说过hello了

如果采用我们的代理对象【占位】显然实现起来非常容易,于是我们首先来生成这个【占位】

  1. import java.lang.reflect.InvocationHandler;
  2. import java.lang.reflect.Method;
  3. import java.lang.reflect.Proxy;
  4. public class HelloServiceProxy implements InvocationHandler {
  5. private Object target;
  6. /**
  7. * 绑定委托对象并返回一个【代理占位】
  8. * @param target 真实对象
  9. * @return  代理对象【占位】
  10. */
  11. public  Object bind(Object target, Class[] interfaces) {
  12. this.target = target;
  13. //取得代理对象
  14. <span style="color:#FF0000;"> return Proxy.newProxyInstance(target.getClass().getClassLoader(),
  15. target.getClass().getInterfaces(), this);  </span>
  16. }
  17. @Override
  18. /**
  19. * 同过代理对象调用方法首先进入这个方法.
  20. * @param proxy --代理对象
  21. * @param method -- 方法,被调用方法.
  22. * @param args -- 方法的参数
  23. */
  24. public Object invoke(Object proxy , Method method, Object[] args) throws Throwable {
  25. System.err.println("############我是JDK动态代理################");
  26. Object result = null;
  27. //反射方法前调用
  28. System.err.println("我准备说hello。");
  29. //反射执行方法  相当于调用target.sayHelllo;
  30. result=method.invoke(target, args);
  31. //反射方法后调用.
  32. System.err.println("我说过hello了");
  33. return result;
  34. }
  35. }

这里的bind方法里面有句:

  1. Proxy.newProxyInstance(target.getClass().getClassLoader(),
  2. target.getClass().getInterfaces(), this);

它的意义就是生成一个代理对象,这里有三个参数:第一个参数是类加载器,第二个参数是或者对象的接口(代理对象挂在那个接口下),第三个参数this代表当前HelloServiceProxy类,换句话说是使用HelloServiceProxy作为对象的代理。

一旦这样绑定后,那么在进入代理对象方法调用的时候就会到HelloServiceProxy的invoke方法上,invoke方法有三个参数:第一个proxy是代理对象,第二个是当前调用那个方法,第三个是方法的参数。

比方说,现在HelloServiceImpl对象(obj)用bind方法绑定后,返回其【占位】(proxy),我们在调用proxy.sa6yHello("张三"),那么它就会进入到HelloServiceProxy的invoke()方法。而invoke参数中第一个便是代理对象proxy,方法便是sayHello,参数是"张三"。那么我们就可以通过反射技术调度真实对象的方法。

现在让我们看看测试代码:

  1. public class ProxyTest {
  2. public static void main(String[] args) {
  3. HelloServiceProxy proxy = new HelloServiceProxy();
  4. HelloService service = new HelloServiceImpl();
  5. //绑定代理对象。
  6. service = (HelloService) proxy.bind(service, new Class[] {HelloService.class});
  7. //这里service经过绑定,就会进入invoke方法里面了。
  8. service.sayHello("张三");
  9. }
  10. }

好,我们运行一下,看一下结果。

  1. ############我是JDK动态代理################
  2. 我准备说hello。
  3. hello 张三
  4. 我说过hello了

好了,我们这样就可以看到动态代理。

正确的理解反射和动态代理是我们进行MyBATIS底层的基础,尤其是动态代理,如果还不了解务必要多做上面的例子,切实理解好。

本文转自:http://blog.csdn.net/ykzhen2015/article/details/50312651

jdk动态代理(转)的更多相关文章

  1. JDK动态代理

    一.基本概念 1.什么是代理? 在阐述JDK动态代理之前,我们很有必要先来弄明白代理的概念.代理这个词本身并不是计算机专用术语,它是生活中一个常用的概念.这里引用维基百科上的一句话对代理进行定义: A ...

  2. 静态代理和利用反射形成的动态代理(JDK动态代理)

    代理模式 代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 静态代理 1.新建 ...

  3. Spring中的JDK动态代理

    Spring中的JDK动态代理 在JDK1.3以后提供了动态代理的技术,允许开发者在运行期创建接口的代理实例.在Sun刚推出动态代理时,还很难想象它有多大的实际用途,现在动态代理是实现AOP的绝好底层 ...

  4. AOP学习心得&jdk动态代理与cglib比较

    什么是AOP AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善.OOP引入 ...

  5. JDK动态代理的实现原理

    学习JDK动态代理,从源码层次来理解其实现原理参考:http://blog.csdn.net/jiankunking/article/details/52143504

  6. Java中的JDK动态代理

    所谓代理,其实就是相当于一个中间人,当客户端需要服务端的服务时,不是客户直接去找服务,而是客户先去找代理,告诉代理需要什么服务,然后代理再去服务端找服务,最后将结果返回给客户. 在日常生活中,就拿买火 ...

  7. JDK动态代理与CGLib动态代理

    1.JDK动态代理 JDK1.3以后java提供了动态代理技术,允许开发者在运行期创建接口的代理实例,动态代理是实现AOP的绝好底层技术. JDK的动态代理主要涉及到java.lang.reflect ...

  8. jdk动态代理实现

    1.jdk动态代理的简单实现类 package com.proxy; import java.lang.reflect.InvocationHandler; import java.lang.refl ...

  9. java jdk动态代理

    在面试的时候面试题里有一道jdk的动态代理是原理,并给一个事例直接写代码出来,现在再整理一下 jdk动态代理主要是想动态在代码中增加一些功能,不影响现有代码,实现动态代理需要做如下几个操作 1.首先必 ...

  10. JDK动态代理的实现及原理

    Proxy.newProxyInstance(classloader,Class,invocationHandler) 调用getProxyClass0(loader, interfaces)生成代理 ...

随机推荐

  1. ExtJs之Ext.XTemplate:模板成员函数

    <!DOCTYPE html> <html> <head> <title>ExtJs</title> <meta http-equiv ...

  2. 解决VTune错误PMU resources currently being used by another profiling tool or process

    错误信息: When I ran Hardware Event-based Sampling Analysis 0, it showed the ERROR: Collection failed Co ...

  3. Qt Quick 之 QML 与 C++ 混合编程具体解释

    Qt Quick 技术的引入.使得你能够高速构建 UI ,具有动画.各种绚丽效果的 UI 都不在话下.但它不是万能的.也有非常多局限性,原来 Qt 的一些技术,比方低阶的网络编程如 QTcpSocke ...

  4. 【智能家居篇】wifi网络结构(上)

    转载请注明出处:http://blog.csdn.net/Righthek 谢谢! WIFI是什么.相信大家都知道,这里就不作说明了. 我们须要做的是深入了解其工作原理,包含软硬件.网络结构等.先说明 ...

  5. 浅析C++绑定到Lua的方法

    注:原文也在公司内部论坛上发了  概述       尽管将C++对象绑定到Lua已经有tolua++(Cocos2d-x 3.0用的就是这个).LuaBridge(我们游戏client对这个库进行了改 ...

  6. Quartz2D二维画图引擎

    Quartz2D二维画图引擎 这个二维画图引擎的功能很强大 一般苹果公司xcode 提供给我们的一些UI控件不能满足我们的需求 所以我们会自己定义控件 xcode 提供的全部控件都是由这个画图引擎画出 ...

  7. 关于wait notify notifyall的学习心得

    wait()能让同步的线程挂起并将锁抛出,sleep只能使线程“睡了“,线程的锁并不会抛出,所以sleep还可以作用于非同步的线程.notify与notifyall能将被挂起或睡着的线程唤醒,但并不是 ...

  8. redis 五大数据类型的常用指令

    STRING 192.168.1.66:6379> get k1 "v1" 192.168.1.66:6379> append k1 12345 (integer) 7 ...

  9. ROS-参数

    前言:参数的用法. 一.参数常用命令 命令 功能 rosparam list 参数列表 rosparam get 获取参数 rosparam set  设置参数 rosparam load 加载参数 ...

  10. POJ 2665 模拟,,

    It is confirmed that these sections do not overlap with each other. 一句话 就变成水题了,,, // by SiriusRen #i ...