八、Dubbo中JavaAssist的Wrapper.getWrapper生成代理分析

一、getWrapper方法入口

private void exportLocal(URL url) {
    if (!Constants.LOCAL_PROTOCOL.equalsIgnoreCase(url.getProtocol())) {
        URL local = URL.valueOf(url.toFullString())
                .setProtocol(Constants.LOCAL_PROTOCOL)
                .setHost(LOCALHOST)
                .setPort(0);
        ServiceClassHolder.getInstance().pushServiceClass(getServiceClass(ref));
        // 這裏會調用proxyFactory的getInvoker方法,proxyFactory是自適應拓展對象,默認是JavassistProxyFactory
        // 因此這裏會走到它的getInvoker方法裏面
        Exporter<?> exporter = protocol.export(
                proxyFactory.getInvoker(ref, (Class) interfaceClass, local));
        exporters.add(exporter);
    }
}
// 咱們先來看一下調用getInvoker時傳遞進來的幾個參數具體是什麼,見下圖
public <T> Invoker<T> getInvoker(T proxy, Class<T> type, URL url) {
    final Wrapper wrapper = 
            Wrapper.getWrapper(proxy.getClass().getName().indexOf('$') < 0 ? proxy.getClass() : type);
            
    return new AbstractProxyInvoker<T>(proxy, type, url) {
        @Override
        protected Object doInvoke(T proxy, String methodName,
                                  Class<?>[] parameterTypes,
                                  Object[] arguments) throws Throwable {
            return wrapper.invokeMethod(proxy, methodName, parameterTypes, arguments);
        }
    };
}

clipboard.png

從圖中能夠看出,傳進來的proxy是DemoServiceImpl的一個實例對象,而type是DemoService接口的class對象,能夠猜想下面的Wrapper.getWrapper方法就是爲該實例對象生成代理對象。下面具體分析。html

二、生成Wrapper類

public static Wrapper getWrapper(Class<?> c) {
    while (ClassGenerator.isDynamicClass(c))
        c = c.getSuperclass();

    if (c == Object.class)
        return OBJECT_WRAPPER;

    // 緩存做用
    Wrapper ret = WRAPPER_MAP.get(c);
    if (ret == null) {
        // 生成Wrapper類,這邊邏輯比較難懂,下面先經過arthas生成具體的Wrapper類觀察一下
        ret = makeWrapper(c);
        WRAPPER_MAP.put(c, ret);
    }
    return ret;
}

上述經過makeWrapper(c)方法會生成具體的Wrapper類,這裏面代碼比較長,它是經過javassist構建 Class。下面先經過阿里開源的arthas工具查看它生成的Wrapper類究竟是什麼? Arthas 用戶文檔。首先啓動Provider服務提供者程序,將斷點放置到ret = makerWrapper(c)這裏,肯定這裏的c是DemoServiceImpl,而後運行程序至下一步,觀察ret的名字。以下圖
clipboard.png
能夠發現生成的Wrapper類是叫Wrapper1,而後放開斷點,讓程序走完,服務提供者處於運行狀態。下圖經過Arthas查看生成的Wrapper1究竟是什麼? 首先下載它的jar包,下載地址。而後打開cmd,執行如下命令。注意2是對應Provider程序的序號,須要和你本身程序對應。輸入2後能夠發現連上了Arthas。而後使用sc命令查找Wrapper1具體對應的是哪一個class文件,最後使用jad命令反編譯,查看Wrapper1類對應的Java文件具體是什麼?
clipboard.png
clipboard.png
最後,咱們能夠獲得Wrapper1的Java文件,以下java

package com.alibaba.dubbo.common.bytecode;
import com.alibaba.dubbo.common.bytecode.ClassGenerator;
import com.alibaba.dubbo.common.bytecode.NoSuchMethodException;
import com.alibaba.dubbo.common.bytecode.NoSuchPropertyException;
import com.alibaba.dubbo.common.bytecode.Wrapper;
import com.alibaba.dubbo.demo.provider.DemoServiceImpl;
import java.lang.reflect.InvocationTargetException;
import java.util.Map;

public class Wrapper1 extends Wrapper implements ClassGenerator.DC {
    public static String[] pns;
    public static Map pts;
    public static String[] mns;
    public static String[] dmns;
    public static Class[] mts0;

    public String[] getPropertyNames() {
        return pns;
    }

    public boolean hasProperty(String string) {
        return pts.containsKey(string);
    }
public Class getPropertyType(String string) {
        return (Class)pts.get(string);
    }

    public String[] getMethodNames() {
        return mns;
    }

    public String[] getDeclaredMethodNames() {
        return dmns;
    }

    public void setPropertyValue(Object object, String string, Object object2) {
        try {
            DemoServiceImpl demoServiceImpl = (DemoServiceImpl)object;
        }
        catch (Throwable throwable) {
            throw new IllegalArgumentException(throwable);
        }
        throw new NoSuchPropertyException(new StringBuffer().append("Not found property 
              \"").append(string).append("\" filed or setter method in class 
              com.alibaba.dubbo.demo.provider.DemoServiceImpl.").toString());
    }
public Object getPropertyValue(Object object, String string) {
        try {
            DemoServiceImpl demoServiceImpl = (DemoServiceImpl)object;
        }
        catch (Throwable throwable) {
            throw new IllegalArgumentException(throwable);
        }
        throw new NoSuchPropertyException(new StringBuffer().append("Not found property 
            \"").append(string).append("\" filed or setter method in class 
            com.alibaba.dubbo.demo.provider.DemoServiceImpl.").toString());
    }
// 這個就是Wrapper對象的invokerMethod方法
    public Object invokeMethod(Object object, String string, Class[] arrclass, 
                                             Object[] arrobject) throws InvocationTargetException {
        DemoServiceImpl demoServiceImpl;
        try {
            demoServiceImpl = (DemoServiceImpl)object;
        }
        catch (Throwable throwable) {
            throw new IllegalArgumentException(throwable);
        }
        try {
            if ("sayHello".equals(string) && arrclass.length == 1) {
                // 調用DemoServiceImpl實例對象的sayHello方法,並將結果返回
                return demoServiceImpl.sayHello((String)arrobject[0]);
            }
        }
        catch (Throwable throwable) {
            throw new InvocationTargetException(throwable);
        }
        throw new NoSuchMethodException(new StringBuffer().append("Not found method \"")
              .append(string)
              .append("\" in class com.alibaba.dubbo.demo.provider.DemoServiceImpl.").toString());
    }
}

如今咱們能夠再來回顧一下JavassistProxyFactory中的getInvoker方法,它首先生成一個Wrapper實例對象,通過分析得知,它其實就是DemoServiceImpl的一個代理對象,內部還有一個invokeMethod方法。getInvoker方法中生成了一個匿名Invoker實例,當外部調用該Invoker的doInvoke方法時,最終會調用wrapper.invokeMethod方法,而wrapper.invokerMethod裏面就是調用的目標對象,即DemoServiceImpl的sayHello方法git

public <T> Invoker<T> getInvoker(T proxy, Class<T> type, URL url) {
    final Wrapper wrapper = 
            Wrapper.getWrapper(proxy.getClass().getName().indexOf('$') < 0 ? proxy.getClass() : type);
            
    return new AbstractProxyInvoker<T>(proxy, type, url) {
        @Override
        protected Object doInvoke(T proxy, String methodName,
                                  Class<?>[] parameterTypes,
                                  Object[] arguments) throws Throwable {
            return wrapper.invokeMethod(proxy, methodName, parameterTypes, arguments);
        }
    };
}

三、makeWrapper方法分析

makeWrapper方法就是建立上述的Wrapper的class文件,閱讀代碼時能夠對照着看,更容易理解,具體就不分析,能夠參考 服務導出github

相關文章
相關標籤/搜索