前段时间,我做了一个艰难的决定,我决定用Go重写了Boxzilla的应用,以前是用PHP的Laravel框架开发的。
在几个星期后,部署了Go应用程序。后来,我测试了一下结果 ,发现比旧的应用有很大的改进,包括更好的性能,更容易部署。
这个应用程序是一个比较简单的数据库驱动的API和用户帐户系统:用户可以登录来下载产品,查看订购的发票,更换付款方式等。
我用Stripe和Braintree两个网关来接受在线支付,发票使用MoneyBird处理,邮件我使用了Mailgun来发送。
当我用Laravel完成项目后,但总觉得有些事情有点复杂。
Laravel每隔一段时间就会发布一个新的版本,告诉我包含了哪些明显的改进,性能又提升了多少倍,会非常好。但很多时候,我觉得它只是对命名和目录结构的更改。
为什么用Go?
去年,我已经把几项服务移到Go上。 但是我还在使用PHP来做产品,比如销售WordPress的二次开发,这个看似古老的技术堆栈,最终卖给最终用户。
如果我是一个自雇者,我会选择一个新的技术来开始我的工作。但是我的老板需要我每天好好工作赚钱。如果条件允许,你想不想有一些新色彩?
而写Go代码就是一种乐趣,它不仅可以快速开发,性能也是快速的。
我想未来几年内,会看到更多人从PHP,Python和JavaScript等动态类型语言切换到Go。
移植代码库
把代码移植到Go语言,主要是数据库的交互,把Laravel中的Blade模板移植到Go中可以使用的东西。
ORM是我比较喜欢的方式,将一些麻烦的数据访问层和纯SQL查询移除。Meddler被用来处理查询结果映射为结构体的模板。
为了像Laravel一样支持多层模板,我用Go封装了一个HTML模板类叫做Grender,现已经开源,它的地址在https://github.com/dannyvankooten/grender。
想在应用中集成Strpe,Strpe官方提供了stripe-go(https://github.com/stripe/stripe-go) 这个包。对于Braintree,有一个叫braintree-go(https://github.com/lionelbarrow/braintree-go)的包,以前更新的不太勤,现在又有了新关注。
另外,Go没有软件包,提供管理Moneybird的发票,我还开源了moneybird-go(https://github.com/dannyvankooten/moneybird-go)。
苹果和桔子的比较
其实Go一个编译型的语言,会好于运行在标准库下解释型的PHP。其实这样比较对PHP是不公平的。但是,我需要分享一些数据,可能会更有趣。
性能
我们使用wrk来测试两个应用的登录页面的性能基准测试。
Concurrency | Avg. latency | Req / sec | Transfer / sec |
|
---|---|---|---|---|
Laravel | 1 | 3.87ms | 261.48 | 1.27MB |
Laravel | 100 | 108.86ms | 917.27 | 6.04MB |
Go | 1 | 325.72μs | 7365.48 | 34.27MB |
Go | 100 | 11.63ms | 19967.31 | 92.91MB |
Go | 200 | 37.68ms | 22653.22 |
105.41MB |
通过上面的表格,我们看到Laravel(或PHP-FPM)在超过了100个并发数据时,性能出现了一些下降。
使用NetData显示出如下图表,可以了解服务器在这些负载时的状态。
Golang有100个并发连接的状态图:
Laravel有100个并发连接的状态图:
请注意,我运行的是与运行应用程序相同的机器的基准测试,所以这极大地影响了这两个图形。
关于代码
我们来比较一下两个应用的代码行数据,包括第三方依赖。
find . -name '*.php' | xargs wc -l
156289 total
Laravel超过156000行代码。这还不包括使用Laravel进行测试的依赖包。
find . -name '*.go' | xargs wc -l 33624 total
另一方面,Golang版本由33.000行代码组成。 这是完全相同功能的代码的五分之一。
我们在Laravel应用程序中排除外部依赖关系,所以我们知道我实际写了多少行。
find . -name '*.php' -not -path "./vendor/*" | xargs wc -l 13921 total
对于Go语言来说:
find . -name '*.go' -not -path "./vendor/*" | xargs wc -l 6750 total
即使是在ide中的代码视图,可能结果会有所增加。不过,相同功能的应用,Go只有一半的代码量。
测试工具
在单元测试中,我们使用是Go提供的citizen类,由以下4个文件组成:
license.go
license_test.go
subscription.go
subscription_test.go
这使得应用测试变得非常方便。
在我们的Laravel应用中,我们主要有集成测试来检查请求处理是否返回正确的响应。 但是PHP的总体测试覆盖率就会比较低,因为紧耦合,在某些时间不能完全覆盖。
小结
其实我做了不应该做的事情,用不同的语言重写一个应用。因为我觉得这样会有更多的乐趣,而且又有小而快的应用程序作为回馈。
扫码二维码 获取免费视频学习资料
- 本文固定链接: http://phpxs.com/post/6874/
- 转载请注明:转载必须在正文中标注并保留原文链接
- 扫码: 扫上方二维码获取免费视频资料