思考并回答以下问题:
在前面几节课,我们使用Golang的net/http库实现了一个带有控制器、路由、中间件等功能的Web框架,凡事都在向完成一个自定义Web框架的方向发展。
在开发的过程中,不知道你有没有意识到,其实框架中的某个模块,比如说路由,实现的方法不止一种,每一个模块要实现的功能也各有不同,所以用哪一种方法来实现,以及要实现哪些功能,都是一种选择。
而每种选择的背后,其实都是方向问题,因为这些选择共同构成了一个框架的倾向性,也就是设计感。你要明白,我们的最终方向是:实现我们想要的理想框架。
这就好比驾驶汽车的时候,作为司机,你要对目的地有明确清晰的认识。那理想框架到底是什么样子的?这个终极问题,闭门造车是无法得到答案的,所以今天,我想让你先从埋头搭建Web框架的视角中暂时跳出来,站在更高的角度来纵观全局。
开源框架怎么比较
我们先进入开源世界,对比开源世界中现有的各种Web框架,理解一下它们的实现选择和意图。
Golang语言的Web开发有很多的框架可以用,但是选择太多也不是好事,因为在技术群里我总会遇到群友有这些疑问:哪款框架比较好呢?我要选择哪款框架呢?这些疑问至少暴露出两个问题:一不知道如何比较开源框架、二不了解这些开源框架,那么接下来我们一一解答。
如何比较开源框架?说到比较,是一定要有标准的。但是衡量一个Web框架是多维度的事情,也没有定论。这里我按照优先级顺序,列出我衡量一个框架的评判标准,你可以参考。
核心模块
一个框架最核心的几个部分就是我们前几节课讲到的:HTTP服务的启动方式、路由、Context、中间件、重启方式,它们的实现非常关键,往往影响到整个框架的表现,也能直接体现出框架作者的开发水平。
框架的核心模块就好像是汽车的引擎,一旦核心模块出了问题,或者有隐性的缺陷,后果往往是无法弥补的。理想的核心模块必须要有设计感,有自己的思想,代码质量、性能都不能出问题。
功能完备性
框架最好能尽可能多地提供功能或者规范,比如有自己的日志模块、脚手架、命令行工具,甚至自己的ORM、缓存等等。
要知道,框架的本质还是在于提高开发效率,在团队中,我们希望不同水平的同学能写出基本一样的代码,那就要靠框架这个顶层设计来规范了。试想一下,如果框架中提供了很方便的参数验证规范,那在开发应用的时候,还有谁愿意走解析和正则匹配来验证参数呢?
在功能完备性的选择上,我们往往会根据之后是否希望自定义需求来确定。这里说的自定义需求,指的是定制自己的日志、ORM等模块的需求。
框架扩展性
理想的框架,它的扩展性一定要好。框架要做的事情应该是定义模块与模块之间的交互标准,而不应该直接定义模块的具体实现方式。我可以在这个标准上扩展出我需要的功能,这样整个框架才会比较灵活。
Web领域的技术边界在不断扩展,谁也无法保证框架中所用的库,哪怕是最基本的日志库,能永远满足需求。当框架使用者想为应用增加一个功能,或者替换某个第三方库的时候,如果改动的地方非常多,要大动干戈,甚至根本就无法支持替换和增加,那这个框架的扩展性就比较弱了。
框架性能
虽然大部分框架都是封装net/http,但是封装程度不同以及具体的实现选择不同(使用的路由匹配、上下文机制等),就会有不同的性能表现。
我们选择框架之后,最终是要在框架上运行代码的,如果运行效率有指数级别的差距,是不能接受的。市面上所有框架的性能,你可以参考这个第三方测评结果。
其实对框架性能来说,大部分场景里,我们是不会有极致性能需求的。所以看各种框架的性能评估,我们不应该把各个框架孤立出来看,应该将差不多量级的性能归为一组。不要去比较单个框架的性能差异,而应该去比较不同量级组之间的性能差异,因为在相同量级下,其实框架和框架之间的性能差别并不是很大。
文档完备度及社区活跃度
开源不仅仅是将代码分享出去,同时分享的还有使用文档,官方必须提供足够的介绍资料,包括文档、视频、demo等。
文档和社区是需要不断运营的,因为在使用的过程中,我们一定会遇到各种各样的问题,官方的回复以及一个活跃的社区是保障问题能得到解决的必要条件。
所以当我们选择一个框架的时候,官网、GitHub的star、issue都是很好的衡量标准。当选择框架之后,不妨每天花一点时间在这个项目的官网、GitHub或者邮件组上,你会得到很多真实信息。
现在基本了解这五点评判标准了,我们可以用这把尺子衡量一下市面上的框架,帮助你迅速熟悉起来。
比较开源框架
在开源社区有个go-web-framework-stars项目,把41款GoWeb框架按流行程度做了个列表,其中Gin、Beego、Echo这三个框架是Star数排名前三的,下面我们就针对这三个框架来分析。
在刚才的五个维度中,我们会重点分析这三个框架的核心模块和功能完备性这两点,另外三点扩展性、性能以及文档完备性资料很多,我只在表格中简单说明一下。
Beego
Beego是一款国人开发的Web框架,它出现得非常早,中间经历过一次比较大的改版,目前版本定位在2.0.0。
先来看Beego的核心模块,我们还是从HTTP服务的启动方式、路由、Context、中间件、重启方式这五个方面来分析。
Beego提供多种服务启动方式:HTTP/HTTPS、CGI、Graceful。
它的路由原理是为每个HTTP方法建立一个路由树,这个路由树结构的每个叶子节点是具体的执行控制器,每个中间节点是HTTP请求中斜杠“/”分隔的节。比如/user/name从根到叶子节点的通路就是:1个根节点、1个中间节点user、一个叶子节点name。
Beego的自动匹配路由和注解路由是比较有特色的,这里我简单解释一下:
- 自动匹配路由的意思是,如果我们把一个控制器注册到自动路由中,路由寻址的时候,会根据“控制器名称+控制器类”中的方法名,自动进行映射匹配。
- 注解路由是指在注册路由树的时候,会根据控制器类中某个方法的注释来解析并创建路由树。
Context的设计方面,Beego的Context是藏在controller结构中的,而不是每个函数的第一个参数带着Context,这个和现在Golang提倡的“函数第一个参数为Context”的规范是有些不同的,可能也是因为Beego出现的时间比较早。
在功能完备性方面,Beego框架提供了很多周边的功能。比如Beego提供了一个原生的ORM框架、自定义的Logger库、参数验证Validate库,甚至还默认提供了管理GC、Goroutine等管理接口。
总体来说,使用Beego的最大感受就是“全”。这是一个很全的框架,开发Web应用所需要的所有组件基本都能在这里找到,如果选择它做业务开发,功能完备性是最重要的因素。所以在从零开始,希望快速开发的场景中,我们会尽量选择使用这个框架。
Echo
Echo框架目前最新版本是v4.0.0。它的底层也是基于net/http,但并没有提供类似Beego通过信号的Graceful启动方式,而是暴露标准库的ShutDown方法进行请求的关闭接口,具体使用的方式是开关还是配置,都交由使用者决定。
在路由方面,Echo的路由也是使用字典树的结构。和Beego的树不同,它并不是为每个HTTP方法建立一个树,而是只整体建立一个树,在叶子节点中,根据不同的HTTP方法存放了不同的控制处理函数,所以你可以在它的叶子节点中看到如下这么一个methodHandler结构:
Echo也封装了自己的Context接口,提供了一系列像JSON、Validate、Bind等很好用的对request和response的封装。
相较于Beego库,Echo库非常轻量,除了基本的路由、Context之外,大部分功能都以Middleware的形式提供出来。Echo的Middleware是一个echo.MiddlewareFunc结构体的无参数函数。在框架的middleware文件夹中,提供了诸如logger、jwt、csrf等中间件。
总体来说,Echo的使用者更看中扩展性和框架性能。和定位一样,它是一种高性能、可扩展、轻量级的Web框架,但麻雀虽小,五脏俱全。目前看起来,它的社区活跃度比Gin框架稍差一些(从两者的中间件贡献数可以看出)。所以我感觉,Echo框架更适合个人开发者,而且需要有一定的扩展框架能力。
Gin
Gin框架目前最新版本1.7.2。它和Echo一样属于非常轻量级的应用,基本实现的就是Web框架最核心的部分。
路由方面,和Echo一样使用字典树,但是又和Echo/Beego不一样的是,它的中间节点并不是斜杠分隔的请求路由,而是相同的字符串部分。
比如下图左边的例子,/welcome和/welgood会创建一个中间节点(/wel)和两个叶子节点(come)和(good),而且并不是只有叶子节点才包含控制器处理函数,中间节点也可能存在控制器节点函数。比如下图的右边例子,/welcome和/welcome2包含中间节点(/welcome)和叶子节点(2),中间节点/welcome也包含了一个处理函数。
Gin的路由还有一个特色就是,使用索引来加速子树查询效率。再看图中左边的例子,/wel中间节点还带着一个索引cg,代表的是子树的第一个字母。所以路由查找的时候,会先去索引cg中查找,当前路由除“/wel”的第一个字母,是否在索引中,如果在,索引的下标序列就代表第几个子树节点。
不过,Gin的路由并不是这个框架作者原创的,使用的是第三方的httprouter包。Gin使用这个核心包的方法也不是直接import第三方包,而是将其copy进自己的代码库,当然可以这么用也是因为httprouter第三方库使用的是BSD这种有很大自由度的许可证协议。
Context基本就和Echo/Beego的设计一样,自己封装了Context结构。但是,Gin的Context结构实现了标准库定义的Context接口,即Deadline、Done等接口。所以按照Golang中定义的鸭子类型概念(长得像鸭子,那么就是鸭子),我们可以把Gin中的Context当作标准库的Context一样使用,这点在实际开发工作中是非常方便的。
在中间件上,Gin框架没有定义所有的middleware,而是定义了middleware函数:1
HandlerFuncfunc(*Context)
链式加载和调用middleware,并将这个middleware的具体实现开放给社区,设计了社区贡献organizations,让社区的开源贡献者把自己实现的中间件统一放在这个organizations中,提供给所有人使用,而Gin的核心代码及相关则放在另一个organizations中。这种为开源社区制定标准,并且不断推进和审核开源贡献代码的做法,也是Gin社区活跃度如此之高的原因之一。
总体来说,Gin比较适合企业级团队使用。它的社区活跃度较高,社区贡献的功能模块较多,能很好补充其功能完备性;同时Gin的扩展性也很好,我们可以在社区贡献模块和自研模块中做出很好的取舍。
这里我们只介绍了三个框架,之后你想要快速了解一个新框架,也可以参考这套思路。
你理想的框架是什么样的
现在你已经知道了如何比较开源框架,也对三款最受欢迎的开源框架有一定了解。那么我们再回答开头提到的终极问题:如果我们要做一款Web框架,它应该是什么样子?
其实从刚才的分析也可以看出来,这个问题是见仁见智的,和你的工作经验、技术阅历、甚至技术的理念都有关系。有的人追求的是世界上运行速度最快的框架;有的人追求的是灵活性最高的框架;有的人追求的是功能最全的框架。
具体你要根据自己的应用场景来选择,包括你Web应用的业务场景、并发需求、团队规模、工期等等。回看刚才提出的五个选择维度,以我们聊过的三个Web框架为例。
在保证框架的核心模块能满足要求的情况下,我们一般在功能完备性和框架扩展性之间取舍。
如果你开发一个运营管理后台,并发量基本在100以下,单机使用,开发的团队规模可能就1~2个人,那你最应该考虑功能完备性,明显使用Beego的收益会远远大于使用Gin和Echo。因为如果你选择Gin和Echo之后,还会遇到比如应该选用哪种ORM、应该选用哪种缓存等一系列问题,而这些在功能组件相当全面的Beego中早就被定义好了。
如果你有一定的团队规模,有比较高的并发量,而且你感觉后续对框架的改动需求或者扩展需求会很高,比如你希望自己开发一个适合团队使用的缓存方法。那么这个时候,你应该把框架扩展性放在最高级,可能Gin和Echo更适合你。如果你要更多的灵活性,你可能会考虑直接从net/http标准库开始,不使用任何的开源Web框架。
所以,选择框架信奉一个原则:只选最适合的,不选最贵的。如果你在几个框架中犹豫不定,除了可以用五个维度比较框架之外,你更应该多花时间内省思考清楚你的真正需求。
小结
我们今天尝试通过回答两个分问题,来思考一个终极问题,理想框架究竟应该是什么样的。
对第一个分问题如何比较开源框架,我们提出了五个维度,按照优先级顺序依次为:核心模块、功能完备性、框架扩展性、框架性能、文档完备度及社区活跃度。然后从这五个维度,简要分析了现在最流行的三个开源框架Beego、Gin和Echo。
最后我们回到终极问题,探讨我们理想中的框架应该是什么样子的?总结一句话就是,在搞清楚真正的业务需求后,选最合适的框架就可以了。把握好这一点,你今后在遇到框架选择问题的时候,就不会太迷茫。
思考题
我们比较了三个框架,但是你可能也用过其他的框架,不妨介绍一下你在过去工作中使用过的框架,以及使用感受?