JAVA代理机制初探

原文地址: http://www.blogjava.net/hadeslee/archive/2007/09/08/143641.html

我对classloader还是不了解,那种Thread.currentThread().getContextClassLoader()的用法还是不懂啊

今天上午写了一个有关于EJB的拦截例子,其实EJB的拦截用的就是JAVA的代理机制.说广一点,EJB的实现就是利用代理实现的远程方法调用.
EJB会在服务器端生成一个实现了所有的接口的类的代理,然后在里面监听你所做的所有事情,并与之反应,这样就实现了远程调用的效果,你在这边调用,而EJB容器在别的地方也可以知道你调用了什么,并返回与之对应的结果,这一切都是用代理来实现的.

下面我们就来认识一下,代理的主要类:java.lang.reflect.Proxy
它定义了一套静态方法,供我们使用,其中一个最常用的方法就是生成代理对象
public static Object newProxyInstance(ClassLoader loader,
                                      Class<?>[] interfaces,
                                      InvocationHandler h)
                               throws IllegalArgumentException

它根据你传入的类加载器和这个代理将会实现的接口,以及一个调用处理器,来生成一个代理对象.说起来比较抽象,还是给点例子吧:
先声明一个接口,用来调用代理的方法

/*
 * MyInterface.java
 *
 * Created on 2007年9月8日, 下午4:38
 *
 * To change this template, choose Tools | Template Manager
 * and open the template in the editor.
 
*/

package  test4;

/**
 *
 * 
@author  hadeslee
 
*/
public   interface  MyInterface {
    
public   void  sayHello(String s);
    
public   void  doSth();
}

然后再写一个类实现此方法
/*
 * Test1.java
 *
 * Created on 2007年9月8日, 下午4:31
 *
 * To change this template, choose Tools | Template Manager
 * and open the template in the editor.
 
*/

package  test4;

import  java.lang.reflect.InvocationHandler;
import  java.lang.reflect.Method;
import  java.lang.reflect.Proxy;
import  java.util.ArrayList;
import  java.util.List;


/**
 *
 * 
@author  hadeslee
 
*/
public   class  Test1  implements  MyInterface{
    
    
/**  Creates a new instance of Test1  */
    
public  Test1() {
        
    }
    
    
public   static   void  main(String[] args)  throws  Exception{
        Test1 list
= new  Test1();
        MyInterface my
= (MyInterface)Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
                list.getClass().getInterfaces(),
                
new  MyHandler < MyInterface > (list));
        System.out.println(
" my.name= " + my.getClass().getName());
        my.doSth();
        my.sayHello(
" 千里冰封 " );
    }
    
// 接口中的方法
     public   void  sayHello(String s) {
        System.out.println(
" sayHello to: " + s);
    }
    
// 接口中的方法
     public   void  doSth() {
        System.out.println(
" doSth() " );
    }
    
// 一个静态内部类,实现了InvocationHandler的接口,
    
// 它也是一个关键的接口,所有代理后的行为都是在这里实现的
     static   class  MyHandler < T >   implements  InvocationHandler{
        
private  T t;
        
public  MyHandler(T t){
            
this .t = t;
        }
        
// 实现方法调用
        
// 可以自己加上自己的一些调用,此例中只是在加上了一个输出
         public  Object invoke(Object proxy, Method method, Object[] args)  throws  Throwable {
            System.out.println(
" 我知道马上要被调用的方法是: " + method.getName());
            
return  method.invoke(t,args);
        }
        
    }
}

运行上面的类输出是:

my.name=$Proxy0
我知道马上要被调用的方法是:doSth
doSth()
我知道马上要被调用的方法是:sayHello
sayHello to:千里冰封

从这里可以看出,代理的类的名字换成了$Proxy0,其中$Proxy是所有代理类的类名前缀
我们在调用doSth()和sayHello()的时候,都调用到了我们在代理中设置的输出.如果你想在这里代理别的类,也是可以的,只要你符合以上的调用规律.
最后特别要注意的一点是:

// 实现方法调用
        
// 可以自己加上自己的一些调用,此例中只是在加上了一个输出
         public  Object invoke(Object proxy, Method method, Object[] args)  throws  Throwable {
            System.out.println(
" 我知道马上要被调用的方法是: " + method.getName());
            
return  method.invoke(t,args);
        }

在上面的实现中,千万不能调用method.invoke(proxy,args).因为proxy本身就是一个代理的对象,你如果再在它上面调用一个方法的话,会无限递归的调用这个方法,所以,在InvocationHandler的实现里面,最好是传一个代理对象的真正实现进去,这样就可以还原本来的调用结果,并加上自己的东西在里面.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值