A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 李蛟龙 中级黑马   /  2012-4-26 10:51  /  1720 人查看  /  4 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

能否谈谈代理的运行机制,
他是不是一个反射的包装的东西?

4 个回复

倒序浏览
在目前的Java开发包中包含了对动态代理的支持,但是其实现只支持对接口的的实现。

  其实现主要通过是java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口。

  Proxy类主要用来获取动态代理对象,InvocationHandler接口用来约束调用者实现,如下,HelloWorld接口定义的业务方法,HelloWorldImpl是HelloWorld接口的实现,HelloWorldHandler是 InvocationHandler接口实现。代码如下:

  业务接口:

         public interface HelloWorld {
  void sayHelloWorld() ;

  }

  业务接口实现:

         public class HelloWorldImpl implements HelloWorld {
  public void sayHelloWorld() {

  System.out.println("Hello World!");

  }

  }

  InvocationHandler实现,需要在接口方法调用前后加入一部份处理工作,这里仅仅在方法调用前后向后台输出两句字符串,其代码如下:

   

      import java.lang.reflect.InvocationHandler;
  import java.lang.reflect.Method;

  public class HelloWorldHandler implements InvocationHandler {

  //要代理的原始对象

  private Object objOriginal;

  /**

  * 构造函数。

  * @param obj 要代理的原始对象。

  */

  public HelloWorldHandler(Object obj) {

  this.objOriginal = obj ;

  }

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

  throws Throwable {

  Object result ;

  //方法调用之前

  doBefore();

  //调用原始对象的方法

  result = method.invoke(this.objOriginal ,args);

  //方法调用之后

  doAfter();

  return result ;

  }

  private void doBefore() {

  System.out.println("before method invoke!");

  }

  private void doAfter() {

  System.out.println("after method invoke!");

  }

  }

测试代码:

         import java.lang.reflect.InvocationHandler;
  import java.lang.reflect.Proxy;

  public class Test {

  public static void main(String[] args) {

  HelloWorld hw = new HelloWorldImpl();

  InvocationHandler handler = new HelloWorldHandler(hw);

  HelloWorld proxy = (HelloWorld) Proxy.newProxyInstance(

  hw.getClass().getClassLoader(),

  hw.getClass().getInterfaces(),

  handler);

  proxy.sayHelloWorld();

  }

  }

  Ø 首先获取一个业务接口的实现对象;

  Ø 获取一个InvocationHandler实现,此处是HelloWorldHandler对象;

  Ø 创建动态代理对象;

  Ø 通过动态代理对象调用sayHelloWorld()方法,此时会在原始对象HelloWorldImpl. sayHelloWorld()方法前后输出两句字符串。

  运行测试类输出如下:

  before method invoke!

  Hello World!

  after method invoke!

  此处Test类中的方法调用代码比较多,在我们的实际应用中可以通过配置文件来来简化客户端的调用实现。另外也可以通过动态代理来实现简单的AOP
回复 使用道具 举报
代理模式中,并不涉及到反射,只是对被代理对象方法的控制。
1,限制开放被代理对象的属性和方法
2,对被代理对象的方法进行附加操作
回复 使用道具 举报

用代理能非常方便的动态的生成目标的代理对象,以代理模式去增强目标的功能。
代理模式的关键点在于代理与目标具有相同的接口,所以创建代理需要给定一个接口。
为了增强目标的功能,创建代理的实例时需要传入增强功能的代码,将增强功能的代码封装在InvocationHandler子类中,传给代理。
代理的运行机制是在对代理实例调用方法时,将对方法调用转给创建时传入的InvocationHandler子类对象的invoke方法,他的声明是:
Object invoke(Object proxy,
              Method method,
              Object[] args)
              throws Throwable
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
回复 使用道具 举报
代理主要是为已开发好的类增加一些功能,比如添加一些测试功能(用户代理的话只需修改配置文件即可不需做太多变动)在交付程序的时候在去掉,这一点在代理视频中也讲过我印象比较深刻,在视频中讲述的是可以用jvm动态生成类,这个类不是代理但可以这么用简化程序员负担,另外这个代理还必须实现一个或多个接口
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马