Serverless给前端带来了什么.docx
《Serverless给前端带来了什么.docx》由会员分享,可在线阅读,更多相关《Serverless给前端带来了什么.docx(13页珍藏版)》请在冰豆网上搜索。
Serverless给前端带来了什么
1.引言
Serverless是一种“无服务器架构”,让用户无需关心程序运行环境、资源及数量,只要将精力Focus到业务逻辑上的技术。
现在公司已经实现DevOps化,正在向Serverless迈进,而为什么前端要关注Serverless?
对业务前端同学:
1.会改变前后端接口定义规范。
2.一定会改变前后端联调方式,让前端参与服务器逻辑开发,甚至NodeJava混部。
3.大大降低Nodejs服务器维护门槛,只要会写JS代码就可以维护Node服务,而无需学习DevOps相关知识。
对一个自由开发者:
4.未来服务器部署更弹性,更省钱。
5.部署速度更快,更不易出错。
前端框架总是带入后端思维,而Serverless则是把前端思维带入了后端运维。
前端开发者其实是最早享受到“Serverless”好处的群体。
他们不需要拥有自己的服务,甚至不需要自己的浏览器,就可以让自己的JS代码均匀、负载均衡的运行在每一个用户的电脑中。
而每个用户的浏览器,就像现在最时髦,最成熟的Serverless集群,从远程加载JS代码开始冷启动,甚至在冷启动上也是卓越领先的:
利用JIT加速让代码实现毫秒级别的冷启动。
不仅如此,浏览器还是实现了BAAS服务的完美环境,我们可以调用任何函数获取用户的Cookie、环境信息、本地数据库服务,而无需关心用户用的是什么电脑,连接了怎样的网络,甚至硬盘的大小。
这就是Serverless理念。
通过FAAS(函数即服务)与BAAS(后台即服务)企图在服务端制造前端开发者习以为常的开发环境,所以前端开发者应该更能理解Serverless带来的好处。
2.精读
FAAS(函数即服务)+BAAS(后台即服务)可以称为一个完整的Serverless的实现,除此之外,还有PASS(平台即服务)的概念。
而通常平台环境都通过容器技术实现,最终都为了达到NoOps(无人运维),或者至少DevOps(开发&运维)。
简单介绍一下这几个名词,防止大家被绕晕:
FAAS-Functionasaservice
函数即服务,每一个函数都是一个服务,函数可以由任何语言编写,除此之外不需要关心任何运维细节,比如:
计算资源、弹性扩容,而且可以按量计费,且支持事件驱动。
业界大云厂商都支持FAAS,各自都有一套工作台、或者可视化工作流来管理这些函数。
BAAS-Backendasaservice
后端及服务,就是集成了许多中间件技术,可以无视环境调用服务,比如数据即服务(数据库服务),缓存服务等。
虽然下面还有很多XAAS,但组成Serverless概念的只有FAAS+BAAS。
PAAS-Platformasaservice
平台即服务,用户只要上传源代码就可以自动持续集成并享受高可用服务,如果速度足够快,可以认为是类似Serverless。
但随着以Docker为代表的容器技术兴起,以容器为粒度的PAAS部署逐渐成为主流,是最常用的应用部署方式。
比如中间件、数据库、操作系统等。
DAAS-Dataasaservice
数据即服务,将数据采集、治理、聚合、服务打包起来提供出去。
DAAS服务可以应用Serverless的架构。
IAAS-InfrastructureasaService
基础设施即服务,比如计算机存储、网络、服务器等基建设施以服务的方式提供。
SAAS-SoftwareasaService
软件即服务,比如ERP、CRM、邮箱服务等,以软件为粒度提供服务。
容器
容器就是隔离了物理环境的虚拟程序执行环境,而且环境可被描述、迁移。
比较热门的容器技术是Docker。
随着容器数量增多,就出现了管理容器集群的技术,比较有名的容器编排平台是Kubernetes。
容器技术是Serverless架构实现的一种选择,也是实现的基础。
NoOps
就是无人运维,比较理想主义,也许要借助AI的能力才能实现完全无人运维。
无人运维不代表Serverless,Serverless可能也需要人运维(至少现在),只是开发者不再需要关心环境。
DevOps
笔者觉得可以理解为“开发即运维”,毕竟出了事情,开发要被问责,而一个成熟的DevOps体系可以让更多的开发者承担OP的职责,或者与OP更密切的合作。
回到Serverless,未来后端开发的体验可能与前端相似:
不需要关心代码运行在哪台服务器(浏览器),无需关心服务器环境(浏览器版本)、不用担心负载均衡(前端从未担心过)、中间件服务随时调用(LocalStorage、ServiceWorker)。
前端同学对Serverless应该尤为激动。
就拿笔者亲身经历举例吧。
从做一款游戏说起
笔者非常迷恋养成类游戏,养成游戏最常见的就是资源建造、收集,或者挂机时计算资源的读秒规则。
笔者在开发游戏的时候,最初是将客户端代码与服务端代码完全分成两套实现的:
//...UI部分,画出一个倒计时伐木场建造进度条
constcurrentTime=awaitrequestBuildingProcess();
constleftTime=newDate().getTime()-currentTime;
//...继续倒计时读条
//读条完毕后,每小时木头产量+100,更新到客户端计时器
store.woodIncrement+=100;
为了游戏体验,用户可以在不刷新浏览器的情况下,看到伐木场建造进度的读条,以及嘭一下建造完毕,并且发现每秒钟多获得了100点木材!
但是当伐木场建造完成前、完成时、完成后的任意时间点刷新浏览器,都要保持逻辑的统一,而且数据需要在后端离线计算。
此时就要写后端代码了:
//每次登陆时,校验当前登陆
constcurrentTime=newDate().getTime()
//获取伐木场当前状态
if(/*建造中*/){
//返回给客户端当前时间
constleftTime=building.startTime-currentTime
res.body=leftTime
}else{
//建造完毕
store.woodIncrement+=100
}
很快,建筑的种类多了起来,不同的状态、等级产量都不同,前后端分开维护成本会越来越大,我们需要做配置同步。
配置同步
为了做前后端配置同步,可以将配置单独托管起来前后端共用,比如新建一个配置文件,专门存储游戏信息:
exportconstbuildings={
wood:
{
name:
"..",
maxLevel:
100,
increamentPerLevel:
50,
initIncreament:
100
}
/*..andsoon..*/
};
这虽然复用了配置,但前后端都有一些共同的逻辑可以复用,比如根据建筑建造时间判断建筑状态,判断N秒后建筑的产量等等。
而Serverless带来了进一步优化的空间。
在Serverless环境下做游戏
试想一下,可以在服务器以函数粒度执行代码,我们可以这样抽象游戏逻辑:
//根据建筑建造时间判断建筑状态
exportconstgetBuildingStatusByTime=(instanceId:
number,time:
number)=>{
/**/
};
//判断建筑生产量
exportconstgetBuildingProduction=(instanceId:
number,lastTime:
number)=>{
conststatus=getBuildingStatusByTime(instanceId,newDate().getTime());
switch(status){
case"building":
return0;
case"finished":
//根据(当前时间-上次打开时间)*每秒产量得到总产量
return;/**/
}
};
//前端UI层,每隔一秒调用一次getBuildingProduction函数,及时更新生产数据
//前端入口函数
exportconstfrontendMain=()=>{
/**/
};
//后端根据每次打开时间,调用一次getBuildingProduction函数并入库
//后端入口函数
exportconstbackendMain=()=>{
/**/
};
利用PASS服务,将前后端逻辑写在一起,将getBuildingProduction函数片段上传至FAAS服务,这样就可以同时共享前后端逻辑了!
在文件夹视图下,可以做如下结构规划:
.
├──client#前端入口
├──server#后端入口
├──common#共享工具函数,可以包含80%的通用游戏逻辑
也许有人会问:
前后端共享代码不止有Serverless才能做到。
的确如此,如果代码抽象足够好,有成熟的工程方案支持,是可以将一份代码分别导出到浏览器与服务器的。
但Serverless基于函数粒度功能更契合前后端复用代码的理念,它的出现可能会推动更广泛的前后端代码复用,这虽然不是新发明,但足够称为一个伟大的改变。
前后端的视角
对于前端开发者,会发现后台服务变简单了。
对于后端开发者,发现服务做厚了,面临的挑战更多了。
更简单的后台服务
传统ECS服务器在租赁时,CentOS与AliyunOS的环境选择就足够让人烦恼。
对个人开发者而言,我们要搭建一个完整的持续集成服务是很困难的,而且面临的选择很多,让人眼花缭乱:
•可以在服务器安装数据库等服务,本地直联服务器的数据库进行开发。
•可以本地安装Docker连接本地数据库服务,将环境打包成镜像整体部署到服务器。
•将前后端代码分离,前端代码在本地开发,服务端代码在服务器开发。
甚至服务器的稳定性,需要PM2等工具进行管理。
当服务器面临攻击、重启、磁盘故障时,打开复杂的工作台或登陆Shell后一通操作才能恢复。
这怎么让人专心把精力放在要做的事情上呢?
Serverless解决了这个问题,因为我们要上传的只是一个代码片段,不再需要面对服务器、系统环境、资源等环境问题,外部服务也有封装好的BAAS体系支持。
实际上在Serverless出来之前,就有许多后端团队利用FAAS理念简化开发流程。
为了减少写后端业务逻辑时,环境、部署问题的干扰,许多团队会将业务逻辑抽象成一个个区块(Block),对应到代码片段或Blockly,这些区块可以独立维护、发布,最后将这些代码片段注入到主程序中,或动态加载。
如果习惯了这种开发方式,那也更容易接受Serverless。
更厚的后台服务
站在后台角度,事情就变得比较复杂了。
相对于提供简单的服务器和容器,现在要对用户屏蔽执行环境,将服务做得更厚。
笔者通过一些文章了解到,Serverless的推行还面临着如下一些挑战:
•Serverless各厂商实现种类很多,想让业务做到多云部署,需要抹平差异。
•成熟的PASS服务其实是伪Serverless,后续该如何标准化。
•FAAS冷启动需要重新加载代码、动态分配资源,导致冷启动速度很慢,除了预热,还需要经济的优化方式。
•对于高并发(比如双11秒杀)场景的应用,无需容量评估是很危险的事情,但如果真能做到完全弹性化,就省去了烦恼的容量评估。
•存量应用如何迁移。
业界的Serverless服务厂商大部分都没有解决存量应用迁移的问题。
•Serverless的特性导致了无状态,而复杂的互联网应用都是有状态的,因此挑战在不改变开发习惯的情况下支持状态。
所幸的是,这些问题都已经在积极处理中,而且不少有了已经落地的解决方案。
Serverless给后台带来的好处远比面临的挑战多:
•推进前后端一体化。
进一步降低Node写服务端代码的门槛,免除应用运营的学习成本。
笔者曾经遇到过自己申请的数据库服务被迁移到其他机房而导致的应用服务中断,以后再也不需要担心了,因为数据库作为BAAS服务,是不需要关心在哪部署,是否跨机房,以及如何做迁移的。
•提高资源利用效率。
杜绝应用独占资源,换成按需加载必然能减少不必要的资源消耗,而且将服务均摊到集群的每一台机器,拉平集群的CPU水位。
•降低云平台使用门槛。
无需运维,灵活拓展,按价值服务,高可用,这些能力在吸引更多客户的同时,完全按需计费的特性也减少了用户开销,达到双赢。
利用Serverless尝试服务开放
笔者在公司负责一个大型BI分析平台建设,BI分析平台的底层能力之一就是可视化搭建。
那么可视化搭建能力该如何开放呢?
现在比较容易做到的是组件开放,毕竟前端可以与后端设计相对解耦,利用AMD加载体系也比较成熟。
现在遇到的一个挑战就是后端能力开放,因为当对取数能力有定制要求时,可能需要定制后端数据处理的逻辑。
目前能做到的是利用maven3、jdk7搭建本地开发环境测试,如果想上线,还需要后端同学的协助。
如果后端搭建一个特有的ServerlessBAAS服务,那么就可以像前端组件一样进行线上Coding,调试,甚至灰度发布进行预发测试。
现在前端云端开发已经有了不少成熟的探索,Serverless可以统一前后端代码在云端开发的体验,而不需要关心环境。
Serverless应用架构设计
看了一些Serverless应用架构图,发现大部分业务都可以套用这样一张架构图:
将业务函数抽象成一个个FAAS函数,将数据库、缓存、加速等服务抽象成BAAS服务。
上层提供Restful或事件触发机制调用,对应到不同的端(PC、移动端)。
想要拓展平台能力,只要在端上做开放(组件接入)与FAAS服务做开放(后端接入)即可。
收益与挑战
Serverless带来了的收益与挑战并存,本文站在前端角度聊一聊。
收益一:
前端更Focus在前端体验技术,而不需要具备太多应用管理知识。
最近看了很多前端前辈写的总结文,最大的体会就是回忆“前端在这几年到底起到了什么作用”。
我们往往会夸大自己的存在感,其实前端存在的意义就是解决人机交互问题,大部分场景下,都是一种锦上添花的作用,而不是必须。
回忆你最自豪的工作经历,可能是掌握了Node应用的运维知识、前端工程体系建设、研发效能优化、标准规范制定等,但真正对业务起效的部分,恰恰是你觉得写得最不值得一提的业务代码。
前端花了太多的时间在周边技术上,而减少了很多对业务、交互的思考。
即便是大公司,也难以招到既熟练使用Nodejs,又具备丰富运维知识的人,同时还要求他前端技术精湛,对业务理解深刻,鱼和熊掌几乎不可兼得。
Serverless可以有效解决这个问题,前端同学只需要会写JS代码而无需掌握任何运维知识,就可以快速实现自己的整套想法。
诚然,了解服务端知识是有必要的,但站在合理分工的角度,前端就应该focus在前端技术上。
前端的核心竞争力或者带来的业务价值,并不会随着了解多一些运维知识而得到补充,相反,这会吞噬掉我们本可以带来更多业务价值的时间。
语言的进化、浏览器的进化、服务器的进化,都是从复杂到简单,底层到封装的过程,而Serverless是后端+运维作为一个整体的进一步封装的过程。
收益二:
逻辑编排带来的代码高度复用、可维护,拓展云+端的能力。
云+端是前端开发的下个形态,提供强大的云编码能力,或者通过插件将端打造为类似云的开发环境。
其最大好处就是屏蔽前端开发环境细节,理念与Serverless类似。
之前有不少团队尝试过利用GraphQL让接口“更有弹性”,而Serverless则是更彻底的方案。
我自己的团队就尝试过GraphQL方案,但由于业务非常复杂,难以用标准的模型描述所有场景的需求,因此不适合使用GraphQL。
恰恰是一套基于Blockly的可视化后端开发平台坚持了下来,而且取得了惊人的开发提效。
这套Blockly通用化抽象后几乎可以由Serverless来代替。
所以Serverless可以解决复杂场景下后端研发提效的问题。
Serverless在融合了云端开发后,就可以通过逻辑编排进一步可视化调整函数执行顺序、依赖关系。
笔者之前在XX广告数据处理团队使用过这种平台计算离线日志,每个MapReduce计算节点经过可视化后,就可以轻松看出故障时哪个节点在阻塞,还可以看到最长执行链路,并为每个节点重新分配执行权重。
即便逻辑编排不能解决开发的所有痛点,但在某个具体业务场景下一定可以大有作为。
挑战一:
Serverless可以完全取消前端转后端的门槛?
前端同学写Node代码最容易犯的毛病就是内存溢出。
浏览器+Tab天然是一个用完即关的场景,UI组件与逻辑创建与销毁也非常频繁,因此前端同学很少,也不太需要关心GC问题。
而GC在后端开发场景中是一个早已养成的习惯,因此Nodejs程序缓存溢出是大家最关注的问题。
Serverless应用是动态加载,长时间不用就会释放的,因此一般来说不需要太担心GC的问题,就算内存溢出,在内存被占满前可能已经进程被释放,或者被监测到异常强制Kill掉。
但毕竟FAAS函数的加载与释放完全是由云端控制的,一个常用的函数长时间不卸载也是有可能的,因此FAAS函数还是要注意控制副作用。
所以Serverless虽然抹平了运维环境,但服务端基本知识还需要了解,必须意识到代码跑在前端还是后端。
挑战二:
性能问题
Serverless的冷启动会导致性能问题,而让业务方主动关心程序的执行频率或者性能要求,再开启预热服务又重新将研发拖入了运维的深渊中。
即便是业界最成熟的亚马逊Serverless云服务,也无法做到业务完全不关心调用频率,就可以轻松应付秒杀场景。
因此目前Serverless可能更适合结合合适的场景使用,而不是任何应用都强行套用Serverless。
虽然可以通过定期运行FAAS服务来保证程序一直Online,但笔者认为这还是违背了Serverless的理念。
挑战三:
如何保证代码可迁移性
有一张很经典的Serverless定位描述图:
网络、存储、服务、虚拟家、操作系统、中间件、运行时、数据都不需要关心了,甚至连应用层都只需要关心其中函数部分,而不需要关心其他比如启动、销毁部分。
前面总拿这点当优势,但也可以反过来认为是个劣势。
当你的代码完全依赖某个公有云环境后,你就失去了整体环境的掌控力,甚至代码都只能在特定的云平台才能运行。
不同云平台提供的BAAS服务规范可能不同,FAAS的入口、执行方式也可能不同,想要采用多云部署就必须克服这个问题。
现在许多Serverless平台都在考虑做标准化,但同时也有一些自下而上的工具库抹平一些差异,比如ServerlessFramework等。
而我们写FAAS函数时,也尽量将与平台绑定的入口函数写得轻一些,将真正的入口放在通用的比如main函数中。
3.总结
Serverless的价值远比挑战大,其理念可以切实解决许多研发效能问题。
但目前Serverless发展阶段仍处于早期,国内的Serverless也处于尝试阶段,而且执行环境存在诸多限制,也就是并没有完全实现Serverless的美好理念,因此如果什么都往上套一定会踩坑。
可能在3-5年后,这些坑会被填平,那么你是选择加入填坑大军,还是选一个合适的场景使用Serverless呢?