dubbo源码怎么看

从去年年中开始,断断续续看了一些dubbo源码。刚开始方式很简单,就是在网上找一些博主的文章跟着一起学,应该说看得很痛苦,说很多东西跟看天书一样也不为过。为什么会有这种感觉?现在回想起来,最大的原因可能是当时对dubbo并没有一个系统的了解,完全不知从何入手,而且网上的很多文章往往只是讲了其中一个点,我并不知道这个东西有什么用,为什么要这么写,以及各模块之间到底是个什么关系。

现在回过头来看,如果当时能有一份类似dubbo学习进阶路线图的资料,也许这源码看起来会轻松很多。

如果现在让我从头开始,我一定不会一开始就闷头扎到源码中去。首先我会从dubbo框架设计入手,从整体设计上对它有个印象。

image

然后具体到源码部分,一定会从SPI入手,了解自适应扩展机制,接下来会看服务导出和服务引用,到这里,基本上hello world的demo你就知道大概是怎么跑起来的了。

当然知其然还要知其所以然,想更深入地了解其运行原理,还必须要了解其协议是怎么设计的,通信过程中的编解码、序列化都做了哪些工作。至此,简单的服务调用过程,你就有了基本的概念。

image

扩展到集群的场景,自然而然,你需要去了解多个生产者如何组成服务目录,消费者如何通过路由和负载均衡来选择生产者,以及相应的集群容错机制。

image

到这里就结束了吗?当然没有。在最新的2.7版本中,dubbo已经形成了三大中心的管理机制,即2.7之前就有的注册中心,以及在2.7中新增的元数据中心和配置中心,在dubbo进阶之路上,这三个中心你当然也需要有所了解。

除此之外,dubbo也一直在演进,在上周参加的dubbo社区开发者大会上,小马哥说将来可能会去掉元数据中心和配置中心(如果我没听错的话),以及在现在很火的云原生方面,dubbo将何去何从?如果你对dubbo感兴趣,这些都是你可以关注的一些话题。

这里给出一张自己简单整理的dubbo学习路径脑图吧。

image

当然,随着去年dubbo进入Apache孵化项目,官方文档上已经有了很多很有价值的资料,里面有很多dubbo commiter以及PMC成员的技术博客,甚至还有dubbo核心模块的源码解析(http://dubbo.apache.org/zh-cn/index.html),我相信这些资料可以帮助初学者更快地入门dubbo框架。

另外,一直以来,我略感困惑的一点是,作为一款优秀的开源框架,为什么市面上一直缺少一本系统介绍dubbo、适合初学者入门和进阶的图书?那么现在,这个困惑可以消除了。因为dubbo PMC成员诣极刚刚出版了这样一本书,我上面提到的一切,这本书里基本都作了详细深入的讲解,干货满满。回想起来,在上一家公司,还跟本书作者诣极大牛有过几次交流,那时候只知道他是dubbo的commiter,如今已是dubbo PMC成员了,还出了这么一本好书,果然,比我优秀的人比我还努力。


image

题外话

坊间一直有一种说法,说dubbo的代码写得不好,跟Spring等优秀的开源框架有不少差距。官方应该也看到这一点了,所以在dubbo重新开始维护之后,除了新特性的开发,官方也对已有功能的代码作了大量的重构和优化。这里以生成自适应扩展类代码的代码来略窥一二。

重构之前的版本是这样的,5+个for循环,15+个if分支,嵌套深度 > 5,单个函数超过300行。

private String createAdaptiveExtensionClassCode() {
...
        for (Method method : methods) {
           ...
            if (adaptiveAnnotation == null) {
                ...
            } else {
                ...
                if (urlTypeIndex != -1) {
                    ...
                }
                else {
                    ...
                    if (attribMethod == null) {
                        ...
                }
                ...
                for (int i = value.length - 1; i >= 0; --i) {
                    if (i == value.length - 1) {
                        if (null != defaultExtName) {
                            if (!"protocol".equals(value[i])) {
                                if (hasInvocation) {
                                    ...
                                } else {
                                    ...
                                }
                            } else {
                               ...
                            }
                        } else {
                            if (!"protocol".equals(value[i])) {
                                if (hasInvocation) {
                                   ...
                                } else {
                                    ...
                                }
                            } else {
                                ...
                            }
                        }
                    } else {
                        if (!"protocol".equals(value[i])) {
                            if (hasInvocation) {
                                ...
                            } else {
                               ...
                            }
                        } else {
                            ...
                        }
                    }
                }
               ...
                for (int i = 0; i < pts.length; i++) {
                    if (i != 0) {
                       ...
                    }
                   ...
                }
               ...
            }
           ...
            for (int i = 0; i < pts.length; i++) {
                if (i > 0) {
                    ...
                }
               ...
            }
            if (ets.length > 0) {
               ...
                for (int i = 0; i < ets.length; i++) {
                    if (i > 0) {
                       ...
                    }
                    ...
                }
            }
           ...
        }
       ...
    }

老实说,当初看到这段代码的时候,我也是一眼蒙圈,不敢相信这真的是dubbo的代码,可以说很好地诠释了如何写出让同事不好维护的代码这一反模式。

好在今年年初,这段代码终于被重构了,重构之后的版本是这样的:

public String generate() {
        // no need to generate adaptive class since there's no adaptive method found.
        if (!hasAdaptiveMethod()) {
            throw new IllegalStateException("No adaptive method exist on extension " + type.getName() + ", refuse to create the adaptive class!");
        }

        StringBuilder code = new StringBuilder();
        code.append(generatePackageInfo());
        code.append(generateImports());
        code.append(generateClassDeclaration());

        Method[] methods = type.getMethods();
        for (Method method : methods) {
            code.append(generateMethod(method));
        }
        code.append("}");

        if (logger.isDebugEnabled()) {
            logger.debug(code.toString());
        }
        return code.toString();
    }


    private String generateMethod(Method method) {
        String methodReturnType = method.getReturnType().getCanonicalName();
        String methodName = method.getName();
        String methodContent = generateMethodContent(method);
        String methodArgs = generateMethodArguments(method);
        String methodThrows = generateMethodThrows(method);
        return String.format(CODE_METHOD_DECLARATION, methodReturnType, methodName, methodArgs, methodThrows, methodContent);
    }

整个过程被抽到了一个单独的类中,并按照类文件的格式被拆分成了十几个短小的方法,基本上每个方法不超过20行,这样的代码层次分明,一目了然,基本上从方法名上你就能知道他要干什么。看了下提交记录,这个重构是在今年2.11(大年初七)提交的,看来作者过年期间也没闲着😆

当然,如果不是下面这段代码的存在,可以说这是一次教科书级别的重构。不过这段代码虽然分支也较多,但好在也只有40行,读起来也不是那么难理解。有意思的是,作者显然也意识到这个问题了,标了个巨大的// TODO: refactor it在这里😆

 private String generateExtNameAssignment(String[] value, boolean hasInvocation) {
        // TODO: refactor it
        String getNameCode = null;
        for (int i = value.length - 1; i >= 0; --i) {
            if (i == value.length - 1) {
                if (null != defaultExtName) {
                    if (!"protocol".equals(value[i])) {
                        if (hasInvocation) {
                           ...
                        } else {
                           ...
                        }
                    } else {
                        ...
                    }
                } else {
                    if (!"protocol".equals(value[i])) {
                        if (hasInvocation) {
                            ...
                        } else {
                            ...
                        }
                    } else {
                       ...
                    }
                }
            } else {
                if (!"protocol".equals(value[i])) {
                    if (hasInvocation) {
                        ...
                       
                    }
                } else {
                    ...
                }
            }
        }

        ...
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 156,907评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,546评论 1 289
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,705评论 0 238
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,624评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,940评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,371评论 1 210
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,672评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,396评论 0 195
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,069评论 1 238
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,350评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,876评论 1 256
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,243评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,847评论 3 231
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,004评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,755评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,378评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,266评论 2 259

推荐阅读更多精彩内容