黑马程序员技术交流社区

标题: 关于动态代理的一些问题 [打印本页]

作者: 王_强    时间: 2013-3-22 10:25
标题: 关于动态代理的一些问题


作者: 孙晋学    时间: 2013-3-22 10:43
代理类我觉得记住来两点吧:第一个是目标类,第二个是功能类
面向AOP的编程,代理是为原来的类(目标)增加新的功能。
Object result = method.invoke(dele, args);这句话是反射,而其中的dele就是目标类,上下两句话(doBefore();和 after(); )就是新增加的功能
目标类和新功能组合出类的新类就是代理类即Proxy.newProxyInstance返回的对象

作者: qintoko    时间: 2013-3-22 10:51
我试图在比较难的代码加上注释
public class LogHandler implements InvocationHandler {//  继承InvocationHandler,重写invoke方法
      
    private Object dele;  
      
    public LogHandler(Object obj) {  
        this.dele = obj;  //传来你要代理的类
    }  
      
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
        doBefore();  
        Object result = method.invoke(dele, args);//反射实现dele方法
        after();  
        return result;  //前后增加方法后返回代理类
    }  
      
    private void doBefore() {  
        System.out.println("before....");  
    }  
      
    private void after() {  
        System.out.println("after....");  
    }  
}


作者: 王_强    时间: 2013-3-22 11:33
qintoko 发表于 2013-3-22 10:51
我试图在比较难的代码加上注释
public class LogHandler implements InvocationHandler {//  继承Invocati ...


Object result = method.invoke(dele, args);//反射实现dele方法
这句话,其实我发现这样写也可以

method.invoke(dele, args);

返回值时:
return null;

也可以正确执行!
为什么很多的程序要那么写呢?不懂?
作者: qintoko    时间: 2013-3-22 11:35
王_强 发表于 2013-3-22 11:33
Object result = method.invoke(dele, args);//反射实现dele方法
这句话,其实我发现这样写也可以

你是要增加after和before的方法才使用代理的,什么都不返回,何必代理呢?直接用原来的类不就行了
作者: 王_强    时间: 2013-3-22 11:36
王_强 发表于 2013-3-22 11:33
Object result = method.invoke(dele, args);//反射实现dele方法
这句话,其实我发现这样写也可以

忘了说谢谢,你的回复,我基本懂了,希望能回复我之前最后一个问题。
作者: 谢洋    时间: 2013-3-22 11:59
下面是个人笔记,希望对你有帮助
/*static Object newProxyInstance(ClassLoader loader, Class<?>[] *interfaces, InvocationHandler h)  
     *正在执行的add()方法:原形
     * add(args){
     * public Object invoke(Object proxy,Method method,Object[] args){
     *         //调用target  的add方法 的代码method.invoke()
     *     }
     * }
     */

生成代理的原理分析:
1、  通过目标实例,得到加载目标实例的类加载器,及目标类的实现的所有接口;
2、  通过InvocationHandler接口产生个子类对象:调用处理器;
3、  并把上面所得到的东西作为参数传给Poxy的静态方法newProxyInstance()
4、  该方法并执行Class cl = getProxyClass(loader, interfaces),得到代理类的字节码
5、  得到代理类指定参数所构造:Constructor cons = cl.getConstructor(constructorParams);
6、  有了构造就可以建立代理实例了:return (Object) cons.newInstance(new Object[] { h });
7、  源代码:上面分析跟源代码差不多吧
private final static Class[] constructorParams =
    { InvocationHandler.class };
public static Object newProxyInstance(ClassLoader loader,
                    Class<?>[] interfaces,
                    InvocationHandler h)
    throws IllegalArgumentException
    {
        if (h == null) {
        throw new NullPointerException();
    }
    /*
     * Look up or generate the designated proxy class.
     */
    Class cl = getProxyClass(loader, interfaces);
    /*
* Invoke its constructor with the designated invocation handler.
     */
    try {
        Constructor cons = cl.getConstructor(constructorParams);
        return (Object) cons.newInstance(new Object[] { h });
    } catch (NoSuchMethodException e) { }
从上面分析及上面源代码可以知道:代理实例只能实现接口的方法,并不是实现了目标类的所有方法。

作者: qintoko    时间: 2013-3-22 13:09
王_强 发表于 2013-3-22 11:36
忘了说谢谢,你的回复,我基本懂了,希望能回复我之前最后一个问题。

为什么很多的程序要那么写呢?不懂?
//是这个问题吗?

这就要牵扯到代理出现的历史渊源,早期版本是没有代理的,如果增加方法有如下途径:
1,修改原始类
2,添加抽象类来继承

但是这两种方式的弊端
1,会造成父类和子类的大量修改 -->出现BUG的不可预计性将会大大增加
2,会造成子类的继承途径发生巨大修改(因为一个子类只能继承一个父类)

在这种情况下,JAVA才出现了代理,目的是方便程序员。

不知道是不是您想要的答案




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2