成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

dubbo源碼解析(二十三)遠(yuǎn)程調(diào)用——Proxy

Cristic / 3170人閱讀

摘要:第二種是,是一款字節(jié)碼引擎工具,能夠在運行時編譯生成。后記該部分相關(guān)的源碼解析地址該文章講解了遠(yuǎn)程調(diào)用中關(guān)于代理的部分,關(guān)鍵部分在于基于實現(xiàn)的字節(jié)碼技術(shù)來支撐動態(tài)代理。

遠(yuǎn)程調(diào)用——Proxy
目標(biāo):介紹遠(yuǎn)程調(diào)用代理的設(shè)計和實現(xiàn),介紹dubbo-rpc-api中的各種proxy包的源碼。
前言

首先聲明叫做代理,代理在很多領(lǐng)域都存在,最形象的就是現(xiàn)在朋友圈的微商代理,廠家委托代理幫他們賣東西。這樣做廠家對于消費者來說就是透明的,并且代理可以自己加上一些活動或者銷售措施,但這并不影響到廠家。這里的廠家就是委托類,而代理就可以抽象為代理類。這樣做有兩個優(yōu)點,第一是可以隱藏代理類的實現(xiàn),第二就是委托類和調(diào)用方的解耦,并且能夠在不修改委托類原本的邏輯情況下新增一些額外的處理。

代理分為兩種,靜態(tài)代理和動態(tài)代理。

靜態(tài)代理:如果代理類在程序運行前就已經(jīng)存在,那么這種代理就是靜態(tài)代理。

動態(tài)代理:代理類在程序運行時創(chuàng)建的代理方式。動態(tài)代理關(guān)系由兩組靜態(tài)代理關(guān)系組成,這就是動態(tài)代理的原理。

上述稍微回顧了一下靜態(tài)代理和動態(tài)代理,那么dubbo對于動態(tài)代理有兩種方法實現(xiàn),分別是javassist和jdk。Proxy 層封裝了所有接口的透明化代理,而在其它層都以 Invoker 為中心,只有到了暴露給用戶使用時,才用 Proxy 將 Invoker 轉(zhuǎn)成接口,或?qū)⒔涌趯崿F(xiàn)轉(zhuǎn)成 Invoker,也就是去掉 Proxy 層 RPC 是可以 Run 的,只是不那么透明,不那么看起來像調(diào)本地服務(wù)一樣調(diào)遠(yuǎn)程服務(wù)。我們來看看下面的圖:

我們能看到左邊是消費者的調(diào)用鏈,只有當(dāng)消費者調(diào)用的時候,ProxyFactory才會通過Proxy把接口實現(xiàn)轉(zhuǎn)化為invoker,并且在其他層的調(diào)用都使用的是invoker,同樣的道理,在服務(wù)提供者暴露服務(wù)的時候,也只有在最后暴露給消費者的時候才會通過Proxy 將 Invoker 轉(zhuǎn)成接口。

動態(tài)代理的底層原理就是字節(jié)碼技術(shù),dubbo提供了兩種方式來實現(xiàn)代理:

第一種jdk,jdk動態(tài)代理比較簡單,它內(nèi)置在JDK中,因此不依賴第三方j(luò)ar包,但是功能相對較弱,當(dāng)調(diào)用Proxy 的靜態(tài)方法創(chuàng)建動態(tài)代理類時,類名格式是“$ProxyN”,N代表第 N 次生成的動態(tài)代理類,如果重復(fù)創(chuàng)建動態(tài)代理類會直接返回原先創(chuàng)建的代理類。但是這個以“$ProxyN”命名的類是繼承Proxy類的,并且實現(xiàn)了其所代理的一組接口,這里就出現(xiàn)了它的一個局限性,由于java的類只能單繼承,所以JDK動態(tài)代理僅支持接口代理。

第二種是Javassist,Javassist是一款Java字節(jié)碼引擎工具,能夠在運行時編譯生成class。該方法也是代理的默認(rèn)方法。

源碼分析 (一)AbstractProxyFactory

該類是代理工廠的抽象類,主要處理了一下需要代理的接口,然后把代理getProxy方法抽象出來。

public abstract class AbstractProxyFactory implements ProxyFactory {

    @Override
    public  T getProxy(Invoker invoker) throws RpcException {
        return getProxy(invoker, false);
    }

    @Override
    public  T getProxy(Invoker invoker, boolean generic) throws RpcException {
        Class[] interfaces = null;
        // 獲得需要代理的接口
        String config = invoker.getUrl().getParameter("interfaces");
        if (config != null && config.length() > 0) {
            // 根據(jù)逗號把每個接口分割開
            String[] types = Constants.COMMA_SPLIT_PATTERN.split(config);
            if (types != null && types.length > 0) {
                // 創(chuàng)建接口類型數(shù)組
                interfaces = new Class[types.length + 2];
                // 第一個放invoker的服務(wù)接口
                interfaces[0] = invoker.getInterface();
                // 第二個位置放回聲測試服務(wù)的接口類
                interfaces[1] = EchoService.class;
                // 其他接口循環(huán)放入
                for (int i = 0; i < types.length; i++) {
                    interfaces[i + 1] = ReflectUtils.forName(types[i]);
                }
            }
        }
        // 如果接口為空,就是config為空,則是回聲測試
        if (interfaces == null) {
            interfaces = new Class[]{invoker.getInterface(), EchoService.class};
        }

        // 如果是泛化服務(wù),那么在代理的接口集合中加入泛化服務(wù)類型
        if (!invoker.getInterface().equals(GenericService.class) && generic) {
            int len = interfaces.length;
            Class[] temp = interfaces;
            interfaces = new Class[len + 1];
            System.arraycopy(temp, 0, interfaces, 0, len);
            interfaces[len] = GenericService.class;
        }

        // 獲得代理
        return getProxy(invoker, interfaces);
    }

    public abstract  T getProxy(Invoker invoker, Class[] types);

}

邏輯比較簡單,就是處理了url中攜帶的interfaces的值。

(二)AbstractProxyInvoker

該類實現(xiàn)了Invoker接口,是代理invoker對象的抽象類。

@Override
public Result invoke(Invocation invocation) throws RpcException {
    try {
        // 調(diào)用了抽象方法doInvoke
        return new RpcResult(doInvoke(proxy, invocation.getMethodName(), invocation.getParameterTypes(), invocation.getArguments()));
    } catch (InvocationTargetException e) {
        return new RpcResult(e.getTargetException());
    } catch (Throwable e) {
        throw new RpcException("Failed to invoke remote proxy method " + invocation.getMethodName() + " to " + getUrl() + ", cause: " + e.getMessage(), e);
    }
}

protected abstract Object doInvoke(T proxy, String methodName, Class[] parameterTypes, Object[] arguments) throws Throwable;

該類最關(guān)鍵的就是這兩個方法,一個是invoke方法,調(diào)用了抽象方法doInvoke,另一個則是抽象方法。該方法被子類實現(xiàn)。

(三)InvokerInvocationHandler

該類實現(xiàn)了InvocationHandler接口,動態(tài)代理類都必須要實現(xiàn)InvocationHandler接口,而該類實現(xiàn)的是對于基礎(chǔ)方法不適用rpc調(diào)用,其他方法使用rpc調(diào)用。

public class InvokerInvocationHandler implements InvocationHandler {

    private final Invoker invoker;

    public InvokerInvocationHandler(Invoker handler) {
        this.invoker = handler;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 獲得方法名
        String methodName = method.getName();
        // 獲得參數(shù)類型
        Class[] parameterTypes = method.getParameterTypes();
        // 如果方法參數(shù)類型是object類型,則直接反射調(diào)用
        if (method.getDeclaringClass() == Object.class) {
            return method.invoke(invoker, args);
        }
        // 基礎(chǔ)方法,不使用 RPC 調(diào)用
        if ("toString".equals(methodName) && parameterTypes.length == 0) {
            return invoker.toString();
        }
        if ("hashCode".equals(methodName) && parameterTypes.length == 0) {
            return invoker.hashCode();
        }
        if ("equals".equals(methodName) && parameterTypes.length == 1) {
            return invoker.equals(args[0]);
        }
        // rpc調(diào)用
        return invoker.invoke(new RpcInvocation(method, args)).recreate();
    }

}
(四)StubProxyFactoryWrapper

該類實現(xiàn)了本地存根的邏輯,關(guān)于本地存根的概念和使用在官方文檔中都有詳細(xì)說明。

地址:http://dubbo.apache.org/zh-cn...
public class StubProxyFactoryWrapper implements ProxyFactory {

    private static final Logger LOGGER = LoggerFactory.getLogger(StubProxyFactoryWrapper.class);

    /**
     * 代理工廠
     */
    private final ProxyFactory proxyFactory;

    /**
     * 協(xié)議
     */
    private Protocol protocol;

    public StubProxyFactoryWrapper(ProxyFactory proxyFactory) {
        this.proxyFactory = proxyFactory;
    }

    public void setProtocol(Protocol protocol) {
        this.protocol = protocol;
    }

    @Override
    public  T getProxy(Invoker invoker, boolean generic) throws RpcException {
        return proxyFactory.getProxy(invoker, generic);
    }

    @Override
    @SuppressWarnings({"unchecked", "rawtypes"})
    public  T getProxy(Invoker invoker) throws RpcException {
        // 獲得代理類對象
        T proxy = proxyFactory.getProxy(invoker);
        // 如果不是返回服務(wù)調(diào)用
        if (GenericService.class != invoker.getInterface()) {
            // 獲得stub的配置
            String stub = invoker.getUrl().getParameter(Constants.STUB_KEY, invoker.getUrl().getParameter(Constants.LOCAL_KEY));
            // 如果配置不為空
            if (ConfigUtils.isNotEmpty(stub)) {
                Class serviceType = invoker.getInterface();
                if (ConfigUtils.isDefault(stub)) {
                    // 根據(jù)local和stub來生成stub
                    if (invoker.getUrl().hasParameter(Constants.STUB_KEY)) {
                        stub = serviceType.getName() + "Stub";
                    } else {
                        stub = serviceType.getName() + "Local";
                    }
                }
                try {
                    // 生成stub類
                    Class stubClass = ReflectUtils.forName(stub);
                    if (!serviceType.isAssignableFrom(stubClass)) {
                        throw new IllegalStateException("The stub implementation class " + stubClass.getName() + " not implement interface " + serviceType.getName());
                    }
                    try {
                        // 獲得構(gòu)造方法,該構(gòu)造方法必須是帶有代理的對象的參數(shù)
                        Constructor constructor = ReflectUtils.findConstructor(stubClass, serviceType);
                        // 使用指定的初始化參數(shù)創(chuàng)建和初始化構(gòu)造函數(shù)聲明類的新實例
                        proxy = (T) constructor.newInstance(new Object[]{proxy});
                        //export stub service
                        URL url = invoker.getUrl();
                        if (url.getParameter(Constants.STUB_EVENT_KEY, Constants.DEFAULT_STUB_EVENT)) {
                            url = url.addParameter(Constants.STUB_EVENT_METHODS_KEY, StringUtils.join(Wrapper.getWrapper(proxy.getClass()).getDeclaredMethodNames(), ","));
                            url = url.addParameter(Constants.IS_SERVER_KEY, Boolean.FALSE.toString());
                            try {
                                //  暴露stub服務(wù)
                                export(proxy, (Class) invoker.getInterface(), url);
                            } catch (Exception e) {
                                LOGGER.error("export a stub service error.", e);
                            }
                        }
                    } catch (NoSuchMethodException e) {
                        throw new IllegalStateException("No such constructor "public " + stubClass.getSimpleName() + "(" + serviceType.getName() + ")" in stub implementation class " + stubClass.getName(), e);
                    }
                } catch (Throwable t) {
                    LOGGER.error("Failed to create stub implementation class " + stub + " in consumer " + NetUtils.getLocalHost() + " use dubbo version " + Version.getVersion() + ", cause: " + t.getMessage(), t);
                    // ignore
                }
            }
        }
        return proxy;
    }

    @Override
    public  Invoker getInvoker(T proxy, Class type, URL url) throws RpcException {
        return proxyFactory.getInvoker(proxy, type, url);
    }

    private  Exporter export(T instance, Class type, URL url) {
        return protocol.export(proxyFactory.getInvoker(instance, type, url));
    }

該類里面最重要的就是getProxy方法的實現(xiàn),在該方法中先根據(jù)配置生成加載stub服務(wù)類,然后通過構(gòu)造方法將代理的對象進(jìn)行包裝,最后暴露該服務(wù),然后返回代理類對象。

(五)JdkProxyFactory

該類繼承了AbstractProxyFactory,是jdk的代理工廠的主要邏輯。

public class JdkProxyFactory extends AbstractProxyFactory {

    @Override
    @SuppressWarnings("unchecked")
    public  T getProxy(Invoker invoker, Class[] interfaces) {
        // 調(diào)用了 Proxy.newProxyInstance直接獲得代理類
        return (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(), interfaces, new InvokerInvocationHandler(invoker));
    }

    @Override
    public  Invoker getInvoker(T proxy, Class type, URL url) {
        // 創(chuàng)建AbstractProxyInvoker對象
        return new AbstractProxyInvoker(proxy, type, url) {
            @Override
            protected Object doInvoke(T proxy, String methodName,
                                      Class[] parameterTypes,
                                      Object[] arguments) throws Throwable {
                // 反射獲得方法
                Method method = proxy.getClass().getMethod(methodName, parameterTypes);
                // 執(zhí)行方法
                return method.invoke(proxy, arguments);
            }
        };
    }

}

不過邏輯實現(xiàn)比較簡單,因為jdk中都封裝好了,直接調(diào)用Proxy.newProxyInstance方法就可以獲得代理類。

(六)JavassistProxyFactory

該類是基于Javassist實現(xiàn)的動態(tài)代理工廠類。

public class JavassistProxyFactory extends AbstractProxyFactory {

    @Override
    @SuppressWarnings("unchecked")
    public  T getProxy(Invoker invoker, Class[] interfaces) {
        // 創(chuàng)建代理
        return (T) Proxy.getProxy(interfaces).newInstance(new InvokerInvocationHandler(invoker));
    }

    @Override
    public  Invoker getInvoker(T proxy, Class type, URL url) {
        // TODO Wrapper cannot handle this scenario correctly: the classname contains "$"
        // 創(chuàng)建Wrapper對象
        final Wrapper wrapper = Wrapper.getWrapper(proxy.getClass().getName().indexOf("$") < 0 ? proxy.getClass() : type);
        return new AbstractProxyInvoker(proxy, type, url) {
            @Override
            protected Object doInvoke(T proxy, String methodName,
                                      Class[] parameterTypes,
                                      Object[] arguments) throws Throwable {
                // 調(diào)用方法
                return wrapper.invokeMethod(proxy, methodName, parameterTypes, arguments);
            }
        };
    }

}

在這里看不出什么具體的實現(xiàn),感覺看起來跟JdkProxyFactory差不多,下面我將講解com.alibaba.dubbo.common.bytecode.Proxy類的getProxy方法和com.alibaba.dubbo.common.bytecode.Wrapper類的getWrapper方法。

(七)Proxy#getProxy()
public static Proxy getProxy(Class... ics) {
    // 獲得代理類
    return getProxy(ClassHelper.getClassLoader(Proxy.class), ics);
}

/**
 * Get proxy.
 *
 * @param cl  class loader.
 * @param ics interface class array.
 * @return Proxy instance.
 */
public static Proxy getProxy(ClassLoader cl, Class... ics) {
    // 最大的代理接口數(shù)限制是65535
    if (ics.length > 65535)
        throw new IllegalArgumentException("interface limit exceeded");

    StringBuilder sb = new StringBuilder();
    // 遍歷代理接口,獲取接口的全限定名并以分號分隔連接成字符串
    for (int i = 0; i < ics.length; i++) {
        // 獲得類名
        String itf = ics[i].getName();
        // 判斷是否為接口
        if (!ics[i].isInterface())
            throw new RuntimeException(itf + " is not a interface.");

        Class tmp = null;
        try {
            // 獲得與itf對應(yīng)的Class對象
            tmp = Class.forName(itf, false, cl);
        } catch (ClassNotFoundException e) {
        }

        // 如果通過類名獲得的類型跟ics中的類型不一樣,則拋出異常
        if (tmp != ics[i])
            throw new IllegalArgumentException(ics[i] + " is not visible from class loader");

        // 拼接類
        sb.append(itf).append(";");
    }

    // use interface class name list as key.
    String key = sb.toString();

    // get cache by class loader.
    Map cache;
    synchronized (ProxyCacheMap) {
        // 通過類加載器獲得緩存
        cache = ProxyCacheMap.get(cl);
        if (cache == null) {
            cache = new HashMap();
            ProxyCacheMap.put(cl, cache);
        }
    }

    Proxy proxy = null;
    synchronized (cache) {
        do {
            Object value = cache.get(key);
            // 如果緩存中存在,則直接返回代理對象
            if (value instanceof Reference) {
                proxy = (Proxy) ((Reference) value).get();
                if (proxy != null)
                    return proxy;
            }

            // 是等待生成的類型,則等待
            if (value == PendingGenerationMarker) {
                try {
                    cache.wait();
                } catch (InterruptedException e) {
                }
            } else {
                // 否則放入緩存中
                cache.put(key, PendingGenerationMarker);
                break;
            }
        }
        while (true);
    }
    // AtomicLong自增生成代理類類名后綴id,防止沖突
    long id = PROXY_CLASS_COUNTER.getAndIncrement();
    String pkg = null;
    ClassGenerator ccp = null, ccm = null;
    try {
        ccp = ClassGenerator.newInstance(cl);

        Set worked = new HashSet();
        List methods = new ArrayList();

        for (int i = 0; i < ics.length; i++) {
            // 判斷是否為public
            if (!Modifier.isPublic(ics[i].getModifiers())) {
                // 獲得該類的包名
                String npkg = ics[i].getPackage().getName();
                if (pkg == null) {
                    pkg = npkg;
                } else {
                    if (!pkg.equals(npkg))
                        throw new IllegalArgumentException("non-public interfaces from different packages");
                }
            }
            // 把接口加入到ccp的mInterfaces中
            ccp.addInterface(ics[i]);

            // 遍歷每個類的方法
            for (Method method : ics[i].getMethods()) {
                // 獲得方法描述 這個方法描述是自定義:
                // 例如:int do(int arg1) => "do(I)I"
                // 例如:void do(String arg1,boolean arg2) => "do(Ljava/lang/String;Z)V"
                String desc = ReflectUtils.getDesc(method);
                if (worked.contains(desc))
                    continue;
                // 如果集合中不存在,則加入該描述
                worked.add(desc);

                int ix = methods.size();
                // 獲得方法返回類型
                Class rt = method.getReturnType();
                // 獲得方法參數(shù)類型
                Class[] pts = method.getParameterTypes();

                // 新建一句代碼
                // 例如Object[] args = new Object[參數(shù)數(shù)量】
                StringBuilder code = new StringBuilder("Object[] args = new Object[").append(pts.length).append("];");
                // 每一個參數(shù)都生成一句代碼
                // 例如args[0] = ($w)$1;
                // 例如 Object ret = handler.invoke(this, methods[3], args);
                for (int j = 0; j < pts.length; j++)
                    code.append(" args[").append(j).append("] = ($w)$").append(j + 1).append(";");
                code.append(" Object ret = handler.invoke(this, methods[" + ix + "], args);");
                // 如果方法不是void類型
                // 則拼接 return ret;
                if (!Void.TYPE.equals(rt))
                    code.append(" return ").append(asArgument(rt, "ret")).append(";");

                methods.add(method);
                ccp.addMethod(method.getName(), method.getModifiers(), rt, pts, method.getExceptionTypes(), code.toString());
            }
        }

        if (pkg == null)
            pkg = PACKAGE_NAME;

        // create ProxyInstance class.
        String pcn = pkg + ".proxy" + id;
        ccp.setClassName(pcn);
        // 添加靜態(tài)字段Method[] methods
        ccp.addField("public static java.lang.reflect.Method[] methods;");
        ccp.addField("private " + InvocationHandler.class.getName() + " handler;");
        // 添加實例對象InvokerInvocationHandler hanler,添加參數(shù)為InvokerInvocationHandler的構(gòu)造器
        ccp.addConstructor(Modifier.PUBLIC, new Class[]{InvocationHandler.class}, new Class[0], "handler=$1;");
        // 添加默認(rèn)無參構(gòu)造器
        ccp.addDefaultConstructor();
        // 使用toClass方法生成對應(yīng)的字節(jié)碼
        Class clazz = ccp.toClass();
        clazz.getField("methods").set(null, methods.toArray(new Method[0]));

        // create Proxy class.
        // 生成的字節(jié)碼對象為服務(wù)接口的代理對象
        String fcn = Proxy.class.getName() + id;
        ccm = ClassGenerator.newInstance(cl);
        ccm.setClassName(fcn);
        ccm.addDefaultConstructor();
        ccm.setSuperClass(Proxy.class);
        ccm.addMethod("public Object newInstance(" + InvocationHandler.class.getName() + " h){ return new " + pcn + "($1); }");
        Class pc = ccm.toClass();
        proxy = (Proxy) pc.newInstance();
    } catch (RuntimeException e) {
        throw e;
    } catch (Exception e) {
        throw new RuntimeException(e.getMessage(), e);
    } finally {
        // release ClassGenerator
        // 重置類構(gòu)造器
        if (ccp != null)
            ccp.release();
        if (ccm != null)
            ccm.release();
        synchronized (cache) {
            if (proxy == null)
                cache.remove(key);
            else
                cache.put(key, new WeakReference(proxy));
            cache.notifyAll();
        }
    }
    return proxy;
}

Proxy是是生成代理對象的工具類,跟JdkProxyFactory中用到的Proxy不是同一個,JdkProxyFactory中的是jdk自帶的java.lang.reflect.Proxy。而該Proxy是dubbo基于javassit實現(xiàn)的com.alibaba.dubbo.common.bytecode.Proxy。該方法比較長,可以分開五個步驟來看:

遍歷代理接口,獲取接口的全限定名,并以分號分隔連接成字符串,以此字符串為key,查找緩存map,如果緩存存在,則獲取代理對象直接返回。

由一個AtomicLong自增生成代理類類名后綴id,防止沖突

遍歷接口中的方法,獲取返回類型和參數(shù)類型,構(gòu)建的方法體見注釋

創(chuàng)建工具類ClassGenerator實例,添加靜態(tài)字段Method[] methods,添加實例對象InvokerInvocationHandler hanler,添加參數(shù)為InvokerInvocationHandler的構(gòu)造器,添加無參構(gòu)造器,然后使用toClass方法生成對應(yīng)的字節(jié)碼。

4中生成的字節(jié)碼對象為服務(wù)接口的代理對象,而Proxy類本身是抽象類,需要實現(xiàn)newInstance(InvocationHandler handler)方法,生成Proxy的實現(xiàn)類,其中proxy0即上面生成的服務(wù)接口的代理對象。

(八)Wrapper#getWrapper
public static Wrapper getWrapper(Class c) {
    // 判斷c是否繼承 ClassGenerator.DC.class ,如果是,則拿到父類,避免重復(fù)包裝
    while (ClassGenerator.isDynamicClass(c)) // can not wrapper on dynamic class.
        c = c.getSuperclass();

    // 如果類為object類型
    if (c == Object.class)
        return OBJECT_WRAPPER;

    // 如果緩存里面沒有該對象,則新建一個wrapper
    Wrapper ret = WRAPPER_MAP.get(c);
    if (ret == null) {
        ret = makeWrapper(c);
        WRAPPER_MAP.put(c, ret);
    }
    return ret;
}

private static Wrapper makeWrapper(Class c) {
    // 如果c不是似有類,則拋出異常
    if (c.isPrimitive())
        throw new IllegalArgumentException("Can not create wrapper for primitive type: " + c);

    // 獲得類名
    String name = c.getName();
    // 獲得類加載器
    ClassLoader cl = ClassHelper.getClassLoader(c);

    // 設(shè)置屬性的方法第一行public void setPropertyValue(Object o, String n, Object v){
    StringBuilder c1 = new StringBuilder("public void setPropertyValue(Object o, String n, Object v){ ");
    // 獲得屬性的方法第一行 public Object getPropertyValue(Object o, String n){
    StringBuilder c2 = new StringBuilder("public Object getPropertyValue(Object o, String n){ ");
    // 執(zhí)行方法的第一行
    StringBuilder c3 = new StringBuilder("public Object invokeMethod(Object o, String n, Class[] p, Object[] v) throws " + InvocationTargetException.class.getName() + "{ ");

    // 添加每個方法中被調(diào)用對象的類型轉(zhuǎn)換的代碼
    c1.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");
    c2.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");
    c3.append(name).append(" w; try{ w = ((").append(name).append(")$1); }catch(Throwable e){ throw new IllegalArgumentException(e); }");

    Map> pts = new HashMap>(); // 
    Map ms = new LinkedHashMap(); // 
    List mns = new ArrayList(); // method names.
    List dmns = new ArrayList(); // declaring method names.

    // get all public field.
    // 遍歷每個public的屬性,放入setPropertyValue和getPropertyValue方法中
    for (Field f : c.getFields()) {
        String fn = f.getName();
        Class ft = f.getType();
        // // 排除有static 和 transient修飾的屬性
        if (Modifier.isStatic(f.getModifiers()) || Modifier.isTransient(f.getModifiers()))
            continue;

        c1.append(" if( $2.equals("").append(fn).append("") ){ w.").append(fn).append("=").append(arg(ft, "$3")).append("; return; }");
        c2.append(" if( $2.equals("").append(fn).append("") ){ return ($w)w.").append(fn).append("; }");
        pts.put(fn, ft);
    }

    Method[] methods = c.getMethods();
    // get all public method.
    boolean hasMethod = hasMethods(methods);
    // 在invokeMethod方法中添加try的代碼
    if (hasMethod) {
        c3.append(" try{");
    }
    // 遍歷方法
    for (Method m : methods) {
        // 忽律Object的方法
        if (m.getDeclaringClass() == Object.class) //ignore Object"s method.
            continue;

        // 判斷方法名和方法參數(shù)長度
        String mn = m.getName();
        c3.append(" if( "").append(mn).append("".equals( $2 ) ");
        // 方法參數(shù)長度
        int len = m.getParameterTypes().length;
        // 判斷方法參數(shù)長度代碼
        c3.append(" && ").append(" $3.length == ").append(len);

        // 若相同方法名存在多個,增加參數(shù)類型數(shù)組的比較判斷
        boolean override = false;
        for (Method m2 : methods) {
            if (m != m2 && m.getName().equals(m2.getName())) {
                override = true;
                break;
            }
        }
        if (override) {
            if (len > 0) {
                for (int l = 0; l < len; l++) {
                    c3.append(" && ").append(" $3[").append(l).append("].getName().equals("")
                            .append(m.getParameterTypes()[l].getName()).append("")");
                }
            }
        }

        c3.append(" ) { ");

        // 如果返回類型是void,則return null,如果不是,則返回對應(yīng)參數(shù)類型
        if (m.getReturnType() == Void.TYPE)
            c3.append(" w.").append(mn).append("(").append(args(m.getParameterTypes(), "$4")).append(");").append(" return null;");
        else
            c3.append(" return ($w)w.").append(mn).append("(").append(args(m.getParameterTypes(), "$4")).append(");");

        c3.append(" }");

        mns.add(mn);
        if (m.getDeclaringClass() == c)
            dmns.add(mn);
        ms.put(ReflectUtils.getDesc(m), m);
    }
    if (hasMethod) {
        c3.append(" } catch(Throwable e) { ");
        c3.append("     throw new java.lang.reflect.InvocationTargetException(e); ");
        c3.append(" }");
    }

    c3.append(" throw new " + NoSuchMethodException.class.getName() + "("Not found method ""+$2+"" in class " + c.getName() + "."); }");

    // 處理get set方法
    // deal with get/set method.
    Matcher matcher;
    for (Map.Entry entry : ms.entrySet()) {
        String md = entry.getKey();
        Method method = (Method) entry.getValue();
        if ((matcher = ReflectUtils.GETTER_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            String pn = propertyName(matcher.group(1));
            c2.append(" if( $2.equals("").append(pn).append("") ){ return ($w)w.").append(method.getName()).append("(); }");
            pts.put(pn, method.getReturnType());
        } else if ((matcher = ReflectUtils.IS_HAS_CAN_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            String pn = propertyName(matcher.group(1));
            c2.append(" if( $2.equals("").append(pn).append("") ){ return ($w)w.").append(method.getName()).append("(); }");
            pts.put(pn, method.getReturnType());
        } else if ((matcher = ReflectUtils.SETTER_METHOD_DESC_PATTERN.matcher(md)).matches()) {
            Class pt = method.getParameterTypes()[0];
            String pn = propertyName(matcher.group(1));
            c1.append(" if( $2.equals("").append(pn).append("") ){ w.").append(method.getName()).append("(").append(arg(pt, "$3")).append("); return; }");
            pts.put(pn, pt);
        }
    }
    c1.append(" throw new " + NoSuchPropertyException.class.getName() + "("Not found property ""+$2+"" filed or setter method in class " + c.getName() + "."); }");
    c2.append(" throw new " + NoSuchPropertyException.class.getName() + "("Not found property ""+$2+"" filed or setter method in class " + c.getName() + "."); }");

    // make class
    long id = WRAPPER_CLASS_COUNTER.getAndIncrement();
    ClassGenerator cc = ClassGenerator.newInstance(cl);
    cc.setClassName((Modifier.isPublic(c.getModifiers()) ? Wrapper.class.getName() : c.getName() + "$sw") + id);
    cc.setSuperClass(Wrapper.class);

    // 增加無參構(gòu)造器
    cc.addDefaultConstructor();
    // 添加屬性
    cc.addField("public static String[] pns;"); // property name array.
    cc.addField("public static " + Map.class.getName() + " pts;"); // property type map.
    cc.addField("public static String[] mns;"); // all method name array.
    cc.addField("public static String[] dmns;"); // declared method name array.
    for (int i = 0, len = ms.size(); i < len; i++)
        cc.addField("public static Class[] mts" + i + ";");

    // 添加屬性相關(guān)的方法
    cc.addMethod("public String[] getPropertyNames(){ return pns; }");
    cc.addMethod("public boolean hasProperty(String n){ return pts.containsKey($1); }");
    cc.addMethod("public Class getPropertyType(String n){ return (Class)pts.get($1); }");
    cc.addMethod("public String[] getMethodNames(){ return mns; }");
    cc.addMethod("public String[] getDeclaredMethodNames(){ return dmns; }");
    cc.addMethod(c1.toString());
    cc.addMethod(c2.toString());
    cc.addMethod(c3.toString());

    try {
        // 生成字節(jié)碼
        Class wc = cc.toClass();
        // setup static field.
        // 反射,設(shè)置靜態(tài)變量的值
        wc.getField("pts").set(null, pts);
        wc.getField("pns").set(null, pts.keySet().toArray(new String[0]));
        wc.getField("mns").set(null, mns.toArray(new String[0]));
        wc.getField("dmns").set(null, dmns.toArray(new String[0]));
        int ix = 0;
        for (Method m : ms.values())
            wc.getField("mts" + ix++).set(null, m.getParameterTypes());
        // // 創(chuàng)建對象并且返回
        return (Wrapper) wc.newInstance();
    } catch (RuntimeException e) {
        throw e;
    } catch (Throwable e) {
        throw new RuntimeException(e.getMessage(), e);
    } finally {
        cc.release();
        ms.clear();
        mns.clear();
        dmns.clear();
    }
}

Wrapper是用于創(chuàng)建某個對象的方法調(diào)用的包裝器,利用字節(jié)碼技術(shù)在調(diào)用方法時進(jìn)行編譯相關(guān)方法。其中g(shù)etWrapper就是獲得Wrapper 對象,其中關(guān)鍵的是makeWrapper方法,所以我在上面加上了makeWrapper方法的解釋,其中就是相關(guān)方法的字節(jié)碼生成過程。

后記
該部分相關(guān)的源碼解析地址:https://github.com/CrazyHZM/i...

該文章講解了遠(yuǎn)程調(diào)用中關(guān)于代理的部分,關(guān)鍵部分在于基于javassist實現(xiàn)的字節(jié)碼技術(shù)來支撐動態(tài)代理。接下來我將開始對rpc模塊的dubbo-rpc-dubbo關(guān)于dubbo協(xié)議部分進(jìn)行講解。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/73008.html

相關(guān)文章

  • dubbo源碼解析(四十七)服務(wù)端處理請求過程

    摘要:而存在的意義就是保證請求或響應(yīng)對象可在線程池中被解碼,解碼完成后,就會分發(fā)到的。 2.7大揭秘——服務(wù)端處理請求過程 目標(biāo):從源碼的角度分析服務(wù)端接收到請求后的一系列操作,最終把客戶端需要的值返回。 前言 上一篇講到了消費端發(fā)送請求的過程,該篇就要將服務(wù)端處理請求的過程。也就是當(dāng)服務(wù)端收到請求數(shù)據(jù)包后的一系列處理以及如何返回最終結(jié)果。我們也知道消費端在發(fā)送請求的時候已經(jīng)做了編碼,所以我...

    yzzz 評論0 收藏0
  • dubbo源碼解析(四十八)異步化改造

    摘要:大揭秘異步化改造目標(biāo)從源碼的角度分析的新特性中對于異步化的改造原理??丛创a解析四十六消費端發(fā)送請求過程講到的十四的,在以前的邏輯會直接在方法中根據(jù)配置區(qū)分同步異步單向調(diào)用。改為關(guān)于可以參考源碼解析十遠(yuǎn)程通信層的六。 2.7大揭秘——異步化改造 目標(biāo):從源碼的角度分析2.7的新特性中對于異步化的改造原理。 前言 dubbo中提供了很多類型的協(xié)議,關(guān)于協(xié)議的系列可以查看下面的文章: du...

    lijinke666 評論0 收藏0
  • dubbo源碼解析(四十五)服務(wù)引用過程

    摘要:服務(wù)引用過程目標(biāo)從源碼的角度分析服務(wù)引用過程。并保留服務(wù)提供者的部分配置,比如版本,,時間戳等最后將合并后的配置設(shè)置為查詢字符串中。的可以參考源碼解析二十三遠(yuǎn)程調(diào)用的一的源碼分析。 dubbo服務(wù)引用過程 目標(biāo):從源碼的角度分析服務(wù)引用過程。 前言 前面服務(wù)暴露過程的文章講解到,服務(wù)引用有兩種方式,一種就是直連,也就是直接指定服務(wù)的地址來進(jìn)行引用,這種方式更多的時候被用來做服務(wù)測試,不...

    xiaowugui666 評論0 收藏0
  • dubbo源碼解析(四十六)消費端發(fā)送請求過程

    摘要:可以參考源碼解析二十四遠(yuǎn)程調(diào)用協(xié)議的八。十六的該類也是用了適配器模式,該類主要的作用就是增加了心跳功能,可以參考源碼解析十遠(yuǎn)程通信層的四。二十的可以參考源碼解析十七遠(yuǎn)程通信的一。 2.7大揭秘——消費端發(fā)送請求過程 目標(biāo):從源碼的角度分析一個服務(wù)方法調(diào)用經(jīng)歷怎么樣的磨難以后到達(dá)服務(wù)端。 前言 前一篇文章講到的是引用服務(wù)的過程,引用服務(wù)無非就是創(chuàng)建出一個代理。供消費者調(diào)用服務(wù)的相關(guān)方法。...

    fish 評論0 收藏0
  • dubbo源碼解析(四十三)2.7新特性

    摘要:大揭秘目標(biāo)了解的新特性,以及版本升級的引導(dǎo)。四元數(shù)據(jù)改造我們知道以前的版本只有注冊中心,注冊中心的有數(shù)十個的鍵值對,包含了一個服務(wù)所有的元數(shù)據(jù)。 DUBBO——2.7大揭秘 目標(biāo):了解2.7的新特性,以及版本升級的引導(dǎo)。 前言 我們知道Dubbo在2011年開源,停止更新了一段時間。在2017 年 9 月 7 日,Dubbo 悄悄的在 GitHub 發(fā)布了 2.5.4 版本。隨后,版本...

    qqlcbb 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<