千里冰封
JAVA 浓香四溢
posts - 151,comments - 2801,trackbacks - 0
今天上午写了一个有关于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的实现里面,最好是传一个代理对象的真正实现进去,这样就可以还原本来的调用结果,并加上自己的东西在里面.



尽管千里冰封
依然拥有晴空

你我共同品味JAVA的浓香.
posted on 2007-09-08 18:12 千里冰封 阅读(6515) 评论(11)  编辑  收藏 所属分类: JAVASE

FeedBack:
# re: JAVA代理机制初探
2007-09-09 07:07 | sitinspring
真是不错,这个方式用对地方了能有很大作为.每天看看Blogjava的首页更新都能有所收获,有时还有意外惊喜.  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-09 09:34 | 兵临城下
这其实和Spring的AOP是一个概念,实现方法都是一致的  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-09 12:39 | BeanSoft
呵呵, 确切说 AOP 的底层实现应该是用代理. 基本上大部分的连接池啊啥的都用代理来做了. 方法拦截(Interceptors)本质上也是代理.  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 00:04 | 杨爱友
@BeanSoft
有了spring的 aop不知道方法拦截还有什么必要,AOP完全可以实现方法拦截啊。  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 02:12 | yexiaozi
(1)生成代理对象:
MyInterface my=(MyInterface)Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
list.getClass().getInterfaces(),
new MyHandler<MyInterface>(list));
(2)然后怎么调用 invoke(*,*,*)?
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("我知道马上要被调用的方法是:"+method.getName());
return method.invoke(t,args);
}
楼主,给个答复看???????????????????????????????????????????  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 02:14 | yexiaozi
楼主:你能否把代码调用的流程,帮我解释下(越详细越好),偶刚接触代理,
麻烦了!辛苦了!谢谢了!  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 14:37 | 千里冰封
@yexiaozi
关于invoke的使用方法如下:
invoke(Object obj, Object... args)
对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。
第一个obj表示你要调用哪个对象上的方法,后面的可变参数表示你传入的参数列表
  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 14:38 | 千里冰封
@yexiaozi
生成了代理对象之后,代理对象会自动拦截你对方法的调用,而调用你在InvokcationHandler里面定义的invoke方法,从而实现所有调用的方法对你来说都是可知道的  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-10 16:05 | 黑盒子
盼望后续深入研究~~~~~  回复  更多评论
  
# re: JAVA代理机制初探
2007-09-16 22:29 | yexiaozi
谢谢楼主!好几天没来看了?今天发现楼主回复了!
理解了一点哦!  回复  更多评论
  
# re: JAVA代理机制初探[未登录]
2008-09-22 10:28 | ben
不错,谢谢楼主分享。  回复  更多评论
  

只有注册用户登录后才能发表评论。


网站导航: